3 # $Id: signup.cgi,v 1.10 2001-08-28 16:58:08 ivan Exp $
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
14 use subs qw( print_form print_okay expselect );
17 use CGI::Carp qw(fatalsToBrowser);
18 use HTTP::Headers::UserAgent 2.00;
19 use FS::SignupClient 0.02 qw( signup_info new_customer );
22 #acceptable payment methods
24 #@payby = qw( CARD BILL COMP );
25 #@payby = qw( CARD BILL );
27 @payby = qw( CARD PREPAY );
29 $ieak_file = '/usr/local/freeside/ieak.template';
30 $cck_file = '/usr/local/freeside/cck.template';
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
37 $ieak_template = new Text::Template ( TYPE => 'STRING', SOURCE => $ieak_txt )
38 or die $Text::Template::ERROR;
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
47 $cck_template = new Text::Template ( TYPE => 'STRING', SOURCE => $cck_txt )
48 or die $Text::Template::ERROR;
53 ( $locales, $packages, $pops ) = signup_info();
57 if ( defined $cgi->param('magic') ) {
58 if ( $cgi->param('magic') eq 'process' ) {
60 $cgi->param('state') =~ /^(\w*)( \(([\w ]+)\))? ?\/ ?(\w+)$/
61 or die "Oops, illegal \"state\" param: ". $cgi->param('state');
66 $payby = $cgi->param('payby');
67 $payinfo = $cgi->param( $payby. '_payinfo' );
69 $cgi->param( $payby. '_month' ). '-'. $cgi->param( $payby. '_year' );
70 $payname = $cgi->param( $payby. '_payname' );
72 if ( $invoicing_list = $cgi->param('invoicing_list') ) {
73 $invoicing_list .= ', POST' if $cgi->param('invoicing_list_POST');
75 $invoicing_list = 'POST';
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'),
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'),
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'),
107 die "unrecognized magic: ". $cgi->param('magic');
125 $invoicing_list = '';
134 $referral_custnum = $cgi->param('ref') || '';
140 my $r = qq!<font color="#ff0000">*</font>!;
142 my $self_url = $cgi->self_url;
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>
149 print qq!<FONT SIZE="+1" COLOR="#ff0000">Error: $error</FONT>! if $error;
152 <FORM ACTION="$self_url" METHOD=POST>
153 <INPUT TYPE="hidden" NAME="magic" VALUE="process">
154 <INPUT TYPE="hidden" NAME="ref" VALUE="$referral_custnum">
156 <TABLE BGCOLOR="#c0c0c0" BORDER=0 CELLSPACING=0 WIDTH="100%">
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>
165 <TD ALIGN="right">Company</TD>
166 <TD COLSPAN=5><INPUT TYPE="text" NAME="company" SIZE=70 VALUE="$company"></TD>
169 <TH ALIGN="right">${r}Address</TH>
170 <TD COLSPAN=5><INPUT TYPE="text" NAME="address1" SIZE=70 VALUE="$address1"></TD>
173 <TD ALIGN="right"> </TD>
174 <TD COLSPAN=5><INPUT TYPE="text" NAME="address2" SIZE=70 VALUE="$address2"></TD>
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">
183 foreach ( @{$locales} ) {
185 print " SELECTED" if ( $state eq $_->{'state'}
186 && $county eq $_->{'county'}
187 && $country eq $_->{'country'}
189 print ">", $_->{'state'};
190 print " (",$_->{'county'},")" if $_->{'county'};
191 print " / ", $_->{'country'};
197 <TD><INPUT TYPE="text" NAME="zip" SIZE=10 VALUE="$zip"></TD>
200 <TD ALIGN="right">Day Phone</TD>
201 <TD COLSPAN=5><INPUT TYPE="text" NAME="daytime" VALUE="$daytime" SIZE=18></TD>
204 <TD ALIGN="right">Night Phone</TD>
205 <TD COLSPAN=5><INPUT TYPE="text" NAME="night" VALUE="$night" SIZE=18></TD>
208 <TD ALIGN="right">Fax</TD>
209 <TD COLSPAN=5><INPUT TYPE="text" NAME="fax" VALUE="$fax" SIZE=12></TD>
211 </TABLE>$r required fields<BR>
212 <BR>Billing information<TABLE BGCOLOR="#c0c0c0" BORDER=0 CELLSPACING=0 WIDTH="100%">
216 print qq!<INPUT TYPE="checkbox" NAME="invoicing_list_POST" VALUE="POST"!;
217 my @invoicing_list = split(', ', $invoicing_list );
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 ),
226 <TR><TD>Billing type</TD></TR></TABLE>
227 <TABLE BGCOLOR="#c0c0c0" BORDER=1 WIDTH="100%">
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>!,
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>!,
246 print qq!<TD VALIGN=TOP><INPUT TYPE="radio" NAME="payby" VALUE="$_"!;
248 print qq! CHECKED> $paybychecked{$_}</TD>!;
250 print qq!> $payby{$_}</TD>!;
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%">
259 <TD COLSPAN=2><SELECT NAME="pkgpart"><OPTION VALUE="">(none)
262 foreach my $package ( @{$packages} ) {
263 print qq!<OPTION VALUE="!, $package->{'pkgpart'}, '"';
264 print " SELECTED" if $pkgpart && ( $package->{'pkgpart'} == $pkgpart );
265 print ">", $package->{'pkg'};
272 <TD ALIGN="right">Username</TD>
273 <TD><INPUT TYPE="text" NAME="username" VALUE="$username"></TD>
276 <TD ALIGN="right">Password</TD>
277 <TD><INPUT TYPE="text" NAME="_password" VALUE="$password">
278 (blank to generate)</TD>
281 <TD ALIGN="right">POP</TD>
282 <TD><SELECT NAME="popnum" SIZE=1><OPTION>
285 foreach my $pop ( @{$pops} ) {
286 print qq!<OPTION VALUE="!, $pop->{'popnum'}, '"',
287 ( $popnum && $pop->{'popnum'} == $popnum ) ? ' SELECTED' : '', ">",
288 $pop->{'popnum'}, ": ",
289 $pop->{'city'}, ", ",
291 " (", $pop->{'ac'}, ")/",
299 <BR><BR><INPUT TYPE="submit" VALUE="Signup">
300 </FORM></BODY></HTML>
306 my $user_agent = new HTTP::Headers::UserAgent $ENV{HTTP_USER_AGENT};
308 $cgi->param('username') =~ /^(.+)$/
309 or die "fatal: invalid username got past FS::SignupClient::new_customer";
311 $cgi->param('_password') =~ /^(.+)$/
312 or die "fatal: invalid password got past FS::SignupClient::new_customer";
314 ( $cgi->param('first'). ' '. $cgi->param('last') ) =~ /^(.*)$/
315 or die "fatal: invalid email_name got past FS::SignupCLient::new_customer";
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'} );
323 && $user_agent->platform eq 'ia32'
324 && $user_agent->os =~ /^win/
325 && ($user_agent->browser)[0] eq 'IE'
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'
335 { #send a Netscape config
336 my $cck_data = $cck_template->fill_in();
337 print $cgi->header('application/x-netscape-autoconfigure-dialer-v2'),
340 pack("N", length($1)). $1. pack("N", length($2)). $2;
341 } split(/\n/, $cck_data);
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>
357 foreach $pop ( @{$pops} ) {
358 if ( $pop->{'popnum'} == $popnum ) { return $pop; }
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 );
372 my $return = qq!<SELECT NAME="$prefix!. qq!_month" SIZE="1">!;
374 $return .= "<OPTION";
375 $return .= " SELECTED" if $_ == $m;
378 $return .= qq!</SELECT>/<SELECT NAME="$prefix!. qq!_year" SIZE="1">!;
379 for ( 1999 .. 2037 ) {
380 $return .= "<OPTION";
381 $return .= " SELECTED" if $_ == $y;
384 $return .= "</SELECT>";