customer-to-customer referrals in singup server
[freeside.git] / fs_signup / FS-SignupClient / cgi / signup.cgi
1 #!/usr/bin/perl -Tw
2 #
3 # $Id: signup.cgi,v 1.10 2001-08-28 16:58:08 ivan Exp $
4
5 use strict;
6 use vars qw( @payby $cgi $locales $packages $pops $r $error
7              $last $first $ss $company $address1 $address2 $city $state $county
8              $country $zip $daytime $night $fax $invoicing_list $payby $payinfo
9              $paydate $payname $referral_custnum
10              $pkgpart $username $password $popnum
11              $ieak_file $ieak_template $cck_file $cck_template
12              $ac $exch $loc
13            );
14 use subs qw( print_form print_okay expselect );
15
16 use CGI;
17 use CGI::Carp qw(fatalsToBrowser);
18 use HTTP::Headers::UserAgent 2.00;
19 use FS::SignupClient 0.02 qw( signup_info new_customer );
20 use Text::Template;
21
22 #acceptable payment methods
23 #
24 #@payby = qw( CARD BILL COMP );
25 #@payby = qw( CARD BILL );
26 #@payby = qw( CARD );
27 @payby = qw( CARD PREPAY );
28
29 $ieak_file = '/usr/local/freeside/ieak.template';
30 $cck_file = '/usr/local/freeside/cck.template';
31
32 if ( -e $ieak_file ) {
33   my $ieak_txt = Text::Template::_load_text($ieak_file)
34     or die $Text::Template::ERROR;
35   $ieak_txt =~ /^(.*)$/s; #untaint the template source - it's trusted
36   $ieak_txt = $1;
37   $ieak_template = new Text::Template ( TYPE => 'STRING', SOURCE => $ieak_txt )
38     or die $Text::Template::ERROR;
39 } else {
40   $ieak_template = '';
41 }
42 if ( -e $cck_file ) {
43   my $cck_txt = Text::Template::_load_text($cck_file)
44     or die $Text::Template::ERROR;
45   $cck_txt =~ /^(.*)$/s; #untaint the template source - it's trusted
46   $cck_txt = $1;
47   $cck_template = new Text::Template ( TYPE => 'STRING', SOURCE => $cck_txt )
48     or die $Text::Template::ERROR;
49 } else {
50   $cck_template = '';
51 }
52
53 ( $locales, $packages, $pops ) = signup_info();
54
55 $cgi = new CGI;
56
57 if ( defined $cgi->param('magic') ) {
58   if ( $cgi->param('magic') eq 'process' ) {
59
60     $cgi->param('state') =~ /^(\w*)( \(([\w ]+)\))? ?\/ ?(\w+)$/
61       or die "Oops, illegal \"state\" param: ". $cgi->param('state');
62     $state = $1;
63     $county = $3 || '';
64     $country = $4;
65
66     $payby = $cgi->param('payby');
67     $payinfo = $cgi->param( $payby. '_payinfo' );
68     $paydate =
69       $cgi->param( $payby. '_month' ). '-'. $cgi->param( $payby. '_year' );
70     $payname = $cgi->param( $payby. '_payname' );
71
72     if ( $invoicing_list = $cgi->param('invoicing_list') ) {
73       $invoicing_list .= ', POST' if $cgi->param('invoicing_list_POST');
74     } else {
75       $invoicing_list = 'POST';
76     }
77
78     ( $error = new_customer ( {
79       'last'             => $last             = $cgi->param('last'),
80       'first'            => $first            = $cgi->param('first'),
81       'ss'               => $ss               = $cgi->param('ss'),
82       'company'          => $company          = $cgi->param('company'),
83       'address1'         => $address1         = $cgi->param('address1'),
84       'address2'         => $address2         = $cgi->param('address2'),
85       'city'             => $city             = $cgi->param('city'),
86       'county'           => $county,
87       'state'            => $state,
88       'zip'              => $zip              = $cgi->param('zip'),
89       'country'          => $country,
90       'daytime'          => $daytime          = $cgi->param('daytime'),
91       'night'            => $night            = $cgi->param('night'),
92       'fax'              => $fax              = $cgi->param('fax'),
93       'payby'            => $payby,
94       'payinfo'          => $payinfo,
95       'paydate'          => $paydate,
96       'payname'          => $payname,
97       'invoicing_list'   => $invoicing_list,
98       'referral_custnum' => $referral_custnum = $cgi->param('ref'),
99       'pkgpart'          => $pkgpart          = $cgi->param('pkgpart'),
100       'username'         => $username         = $cgi->param('username'),
101       '_password'        => $password         = $cgi->param('_password'),
102       'popnum'           => $popnum           = $cgi->param('popnum'),
103     } ) )
104       ? print_form()
105       : print_okay();
106   } else {
107     die "unrecognized magic: ". $cgi->param('magic');
108   }
109 } else {
110   $error = '';
111   $last = '';
112   $first = '';
113   $ss = '';
114   $company = '';
115   $address1 = '';
116   $address2 = '';
117   $city = '';
118   $state = '';
119   $county = '';
120   $country = '';
121   $zip = '';
122   $daytime = '';
123   $night = '';
124   $fax = '';
125   $invoicing_list = '';
126   $payby = '';
127   $payinfo = '';
128   $paydate = '';
129   $payname = '';
130   $pkgpart = '';
131   $username = '';
132   $password = '';
133   $popnum = '';
134   $referral_custnum = $cgi->param('ref') || '';
135   print_form;
136 }
137
138 sub print_form {
139
140   my $r = qq!<font color="#ff0000">*</font>!;
141   $cgi->delete('ref');
142   my $self_url = $cgi->self_url;
143
144   print $cgi->header( '-expires' => 'now' ), <<END;
145 <HTML><HEAD><TITLE>ISP Signup form</TITLE></HEAD>
146 <BODY BGCOLOR="#e8e8e8"><FONT SIZE=7>ISP Signup form</FONT><BR><BR>
147 END
148
149   print qq!<FONT SIZE="+1" COLOR="#ff0000">Error: $error</FONT>! if $error;
150
151   print <<END;
152 <FORM ACTION="$self_url" METHOD=POST>
153 <INPUT TYPE="hidden" NAME="magic" VALUE="process">
154 <INPUT TYPE="hidden" NAME="ref" VALUE="$referral_custnum">
155 Contact Information
156 <TABLE BGCOLOR="#c0c0c0" BORDER=0 CELLSPACING=0 WIDTH="100%">
157 <TR>
158   <TH ALIGN="right">${r}Contact name<BR>(last, first)</TH>
159   <TD COLSPAN=3><INPUT TYPE="text" NAME="last" VALUE="$last">,
160                 <INPUT TYPE="text" NAME="first" VALUE="$first"></TD>
161   <TD ALIGN="right">SS#</TD>
162   <TD><INPUT TYPE="text" NAME="ss" SIZE=11 VALUE="$ss"></TD>
163 </TR>
164 <TR>
165   <TD ALIGN="right">Company</TD>
166   <TD COLSPAN=5><INPUT TYPE="text" NAME="company" SIZE=70 VALUE="$company"></TD>
167 </TR>
168 <TR>
169   <TH ALIGN="right">${r}Address</TH>
170   <TD COLSPAN=5><INPUT TYPE="text" NAME="address1" SIZE=70 VALUE="$address1"></TD>
171 </TR>
172 <TR>
173   <TD ALIGN="right">&nbsp;</TD>
174   <TD COLSPAN=5><INPUT TYPE="text" NAME="address2" SIZE=70 VALUE="$address2"></TD>
175 </TR>
176 <TR>
177   <TH ALIGN="right">${r}City</TH>
178   <TD><INPUT TYPE="text" NAME="city" VALUE="$city"></TD>
179   <TH ALIGN="right">${r}State/Country</TH>
180   <TD><SELECT NAME="state" SIZE="1">
181 END
182
183   foreach ( @{$locales} ) {
184     print "<OPTION";
185     print " SELECTED" if ( $state eq $_->{'state'}
186                            && $county eq $_->{'county'}
187                            && $country eq $_->{'country'}
188                          );
189     print ">", $_->{'state'};
190     print " (",$_->{'county'},")" if $_->{'county'};
191     print " / ", $_->{'country'};
192   }
193
194   print <<END;
195   </SELECT></TD>
196   <TH>${r}Zip</TH>
197   <TD><INPUT TYPE="text" NAME="zip" SIZE=10 VALUE="$zip"></TD>
198 </TR>
199 <TR>
200   <TD ALIGN="right">Day Phone</TD>
201   <TD COLSPAN=5><INPUT TYPE="text" NAME="daytime" VALUE="$daytime" SIZE=18></TD>
202 </TR>
203 <TR>
204   <TD ALIGN="right">Night Phone</TD>
205   <TD COLSPAN=5><INPUT TYPE="text" NAME="night" VALUE="$night" SIZE=18></TD>
206 </TR>
207 <TR>
208   <TD ALIGN="right">Fax</TD>
209   <TD COLSPAN=5><INPUT TYPE="text" NAME="fax" VALUE="$fax" SIZE=12></TD>
210 </TR>
211 </TABLE>$r required fields<BR>
212 <BR>Billing information<TABLE BGCOLOR="#c0c0c0" BORDER=0 CELLSPACING=0 WIDTH="100%">
213 <TR><TD>
214 END
215
216   print qq!<INPUT TYPE="checkbox" NAME="invoicing_list_POST" VALUE="POST"!;
217   my @invoicing_list = split(', ', $invoicing_list );
218   print ' CHECKED'
219     if ! @invoicing_list || grep { $_ eq 'POST' } @invoicing_list;
220   print '>Postal mail invoice</TD></TR><TR><TD>Email invoice ',
221          qq!<INPUT TYPE="text" NAME="invoicing_list" VALUE="!,
222          join(', ', grep { $_ ne 'POST' } @invoicing_list ),
223          qq!"></TD></TR>!;
224
225   print <<END;
226 <TR><TD>Billing type</TD></TR></TABLE>
227 <TABLE BGCOLOR="#c0c0c0" BORDER=1 WIDTH="100%">
228 <TR>
229 END
230
231   my %payby = (
232     'CARD' => qq!Credit card<BR>${r}<INPUT TYPE="text" NAME="CARD_payinfo" VALUE="" MAXLENGTH=19><BR>${r}Exp !. expselect("CARD"). qq!<BR>${r}Name on card<BR><INPUT TYPE="text" NAME="CARD_payname" VALUE="">!,
233     'BILL' => qq!Billing<BR>P.O. <INPUT TYPE="text" NAME="BILL_payinfo" VALUE=""><BR>${r}Exp !. expselect("BILL", "12-2037"). qq!<BR>${r}Attention<BR><INPUT TYPE="text" NAME="BILL_payname" VALUE="Accounts Payable">!,
234     'COMP' => qq!Complimentary<BR>${r}Approved by<INPUT TYPE="text" NAME="COMP_payinfo" VALUE=""><BR>${r}Exp !. expselect("COMP"),
235     'PREPAY' => qq!Prepaid card<BR>${r}<INPUT TYPE="text" NAME="PREPAY_payinfo" VALUE="" MAXLENGTH=80>!,
236   );
237
238   my %paybychecked = (
239     'CARD' => qq!Credit card<BR>${r}<INPUT TYPE="text" NAME="CARD_payinfo" VALUE="$payinfo" MAXLENGTH=19><BR>${r}Exp !. expselect("CARD", $paydate). qq!<BR>${r}Name on card<BR><INPUT TYPE="text" NAME="CARD_payname" VALUE="$payname">!,
240     'BILL' => qq!Billing<BR>P.O. <INPUT TYPE="text" NAME="BILL_payinfo" VALUE="$payinfo"><BR>${r}Exp !. expselect("BILL", $paydate). qq!<BR>${r}Attention<BR><INPUT TYPE="text" NAME="BILL_payname" VALUE="$payname">!,
241     'COMP' => qq!Complimentary<BR>${r}Approved by<INPUT TYPE="text" NAME="COMP_payinfo" VALUE="$payinfo"><BR>${r}Exp !. expselect("COMP", $paydate),
242     'PREPAY' => qq!Prepaid card<BR>${r}<INPUT TYPE="text" NAME="PREPAY_payinfo" VALUE="$payinfo" MAXLENGTH=80>!,
243   );
244
245   for (@payby) {
246     print qq!<TD VALIGN=TOP><INPUT TYPE="radio" NAME="payby" VALUE="$_"!;
247     if ($payby eq $_) {
248       print qq! CHECKED> $paybychecked{$_}</TD>!;
249     } else {
250       print qq!> $payby{$_}</TD>!;
251     }
252   }
253
254   print <<END;
255 </TR></TABLE>$r required fields for each billing type
256 <BR><BR>First package
257 <TABLE BGCOLOR="#c0c0c0" BORDER=0 CELLSPACING=0 WIDTH="100%">
258 <TR>
259   <TD COLSPAN=2><SELECT NAME="pkgpart"><OPTION VALUE="">(none)
260 END
261
262   foreach my $package ( @{$packages} ) {
263     print qq!<OPTION VALUE="!, $package->{'pkgpart'}, '"';
264     print " SELECTED" if $pkgpart && ( $package->{'pkgpart'} == $pkgpart );
265     print ">", $package->{'pkg'};
266   }
267
268   print <<END;
269   </SELECT></TD>
270 </TR>
271 <TR>
272   <TD ALIGN="right">Username</TD>
273   <TD><INPUT TYPE="text" NAME="username" VALUE="$username"></TD>
274 </TR>
275 <TR>
276   <TD ALIGN="right">Password</TD>
277   <TD><INPUT TYPE="text" NAME="_password" VALUE="$password">
278   (blank to generate)</TD>
279 </TR>
280 <TR>
281   <TD ALIGN="right">POP</TD>
282   <TD><SELECT NAME="popnum" SIZE=1><OPTION> 
283 END
284
285   foreach my $pop ( @{$pops} ) {
286     print qq!<OPTION VALUE="!, $pop->{'popnum'}, '"',
287           ( $popnum && $pop->{'popnum'} == $popnum ) ? ' SELECTED' : '', ">", 
288           $pop->{'popnum'}, ": ", 
289           $pop->{'city'}, ", ",
290           $pop->{'state'},
291           " (", $pop->{'ac'}, ")/",
292           $pop->{'exch'}, "\n"
293         ;
294   }
295   print <<END;
296   </SELECT></TD>
297 </TR>
298 </TABLE>
299 <BR><BR><INPUT TYPE="submit" VALUE="Signup">
300 </FORM></BODY></HTML>
301 END
302
303 }
304
305 sub print_okay {
306   my $user_agent = new HTTP::Headers::UserAgent $ENV{HTTP_USER_AGENT};
307
308   $cgi->param('username') =~ /^(.+)$/
309     or die "fatal: invalid username got past FS::SignupClient::new_customer";
310   my $username = $1;
311   $cgi->param('_password') =~ /^(.+)$/
312     or die "fatal: invalid password got past FS::SignupClient::new_customer";
313   my $password = $1;
314   ( $cgi->param('first'). ' '. $cgi->param('last') ) =~ /^(.*)$/
315     or die "fatal: invalid email_name got past FS::SignupCLient::new_customer";
316   my $email_name = $1;
317
318   my $pop = pop_info($cgi->param('popnum'))
319     or die "fatal: invalid popnum got past FS::SignupClient::new_customer";
320   ( $ac, $exch, $loc ) = ( $pop->{'ac'}, $pop->{'exch'}, $pop->{'loc'} );
321
322   if ( $ieak_template
323        && $user_agent->platform eq 'ia32'
324        && $user_agent->os =~ /^win/
325        && ($user_agent->browser)[0] eq 'IE'
326      )
327   { #send an IEAK config
328     print $cgi->header('application/x-Internet-signup'),
329           $ieak_template->fill_in();
330   } elsif ( $cck_template
331             && $user_agent->platform eq 'ia32'
332             && $user_agent->os =~ /^win/
333             && ($user_agent->browser)[0] eq 'Netscape'
334           )
335   { #send a Netscape config
336     my $cck_data = $cck_template->fill_in();
337     print $cgi->header('application/x-netscape-autoconfigure-dialer-v2'),
338           map {
339             m/(.*)\s+(.*)$/;
340             pack("N", length($1)). $1. pack("N", length($2)). $2;
341           } split(/\n/, $cck_data);
342
343   } else { #send a simple confirmation
344     print $cgi->header( '-expires' => 'now' ), <<END;
345 <HTML><HEAD><TITLE>Signup successful</TITLE></HEAD>
346 <BODY BGCOLOR="#e8e8e8"><FONT SIZE=7>Signup successful</FONT><BR><BR>
347 blah blah blah
348 </BODY>
349 </HTML>
350 END
351   }
352 }
353
354 sub pop_info {
355   my $popnum = shift;
356   my $pop;
357   foreach $pop ( @{$pops} ) {
358     if ( $pop->{'popnum'} == $popnum ) { return $pop; }
359   }
360   '';
361 }
362
363 sub expselect {
364   my $prefix = shift;
365   my $date = shift || '';
366   my( $m, $y ) = ( 0, 0 );
367   if ( $date  =~ /^(\d{4})-(\d{2})-\d{2}$/ ) { #PostgreSQL date format
368     ( $m, $y ) = ( $2, $1 );
369   } elsif ( $date =~ /^(\d{1,2})-(\d{1,2}-)?(\d{4}$)/ ) {
370     ( $m, $y ) = ( $1, $3 );
371   }
372   my $return = qq!<SELECT NAME="$prefix!. qq!_month" SIZE="1">!;
373   for ( 1 .. 12 ) {
374     $return .= "<OPTION";
375     $return .= " SELECTED" if $_ == $m;
376     $return .= ">$_";
377   }
378   $return .= qq!</SELECT>/<SELECT NAME="$prefix!. qq!_year" SIZE="1">!;
379   for ( 1999 .. 2037 ) {
380     $return .= "<OPTION";
381     $return .= " SELECTED" if $_ == $y;
382     $return .= ">$_";
383   }
384   $return .= "</SELECT>";
385
386   $return;
387 }
388