be6f2f758d9b379233379475f8657eff6bdd841c
[freeside.git] / htetc / handler.pl
1 #!/usr/bin/perl
2
3 package HTML::Mason;
4
5 use strict;
6 use vars qw($r);
7 use HTML::Mason 1.27; #http://www.masonhq.com/?ApacheModPerl2Redirect
8 use HTML::Mason::Interp;
9 use HTML::Mason::Compiler::ToObject;
10
11 # Bring in ApacheHandler, necessary for mod_perl integration.
12 # Uncomment the second line (and comment the first) to use
13 # Apache::Request instead of CGI.pm to parse arguments.
14 use HTML::Mason::ApacheHandler;
15 # use HTML::Mason::ApacheHandler (args_method=>'mod_perl');
16
17 ###use Module::Refresh;###
18
19 # List of modules that you want to use from components (see Admin
20 # manual for details)
21 #{  package HTML::Mason::Commands;
22 #   use CGI;
23 #}
24
25 if ( %%%RT_ENABLED%%% ) {
26  eval '
27    use lib ( "/opt/rt3/local/lib", "/opt/rt3/lib" );
28    use RT;
29    use vars qw($Nobody $SystemUser);
30    RT::LoadConfig();
31  ';
32  die $@ if $@;
33 }
34
35 # Create Mason objects
36
37 my %interp = (
38   request_class        => 'HTML::Mason::Request::ApacheHandler',
39   data_dir             => '%%%MASONDATA%%%',
40   error_mode           => 'output',
41   error_format         => 'html',
42   ignore_warnings_expr => '.',
43   comp_root            => [
44                             [ 'freeside' => '%%%FREESIDE_DOCUMENT_ROOT%%%'    ],
45                             [ 'rt'       => '%%%FREESIDE_DOCUMENT_ROOT%%%/rt' ],
46                           ],
47 );
48
49 my $fs_interp = new HTML::Mason::Interp (
50   %interp,
51   escape_flags => { 'js_string' => sub {
52                       #${$_[0]} =~ s/(['\\\n])/'\\'.($1 eq "\n" ? 'n' : $1)/ge;
53                       ${$_[0]} =~ s/(['\\])/\\$1/g;
54                       ${$_[0]} =~ s/\n/\\n/g;
55                       ${$_[0]} = "'". ${$_[0]}. "'";
56                     }
57                   },
58 );
59
60 my $rt_interp = new HTML::Mason::Interp (
61   %interp,
62   escape_flags => { 'h' => \&RT::Interface::Web::EscapeUTF8 },
63   compiler     => HTML::Mason::Compiler::ToObject->new(
64                     default_escape_flags => 'h',
65                     allow_globals        => [qw(%session)],
66                   ),
67 );
68
69 my $ah = new HTML::Mason::ApacheHandler (
70   interp      => $fs_interp,
71   args_method => 'CGI', #(and FS too)
72 );
73
74 # Activate the following if running httpd as root (the normal case).
75 # Resets ownership of all files created by Mason at startup.
76 #
77 #chown (Apache->server->uid, Apache->server->gid, $interp->files_written);
78
79 sub handler
80 {
81     ($r) = @_;
82
83     # If you plan to intermix images in the same directory as
84     # components, activate the following to prevent Mason from
85     # evaluating image files as components.
86     #
87     #return -1 if $r->content_type && $r->content_type !~ m|^text/|i;
88
89     #rar
90     { package HTML::Mason::Commands;
91       use strict;
92       use vars qw( $cgi $p $fsurl);
93       use vars qw( %session );
94       use CGI 3.29 qw(-private_tempfiles); #3.29 to fix RT attachment problems
95       #use CGI::Carp qw(fatalsToBrowser);
96       use CGI::Cookie;
97       use List::Util qw( max min );
98       use Data::Dumper;
99       use Date::Format;
100       use Date::Parse;
101       use Time::Local;
102       use Time::Duration;
103       use DateTime;
104       use DateTime::Format::Strptime;
105       use Lingua::EN::Inflect qw(PL);
106       use Tie::IxHash;
107       use URI::URL;
108       use URI::Escape;
109       use HTML::Entities;
110       use HTML::TreeBuilder;
111       use HTML::FormatText;
112       use JSON;
113       use MIME::Base64;
114       use IO::Handle;
115       use IO::File;
116       use IO::Scalar;
117       #not actually using this yet anyway...# use IPC::Run3 0.036;
118       use Net::Whois::Raw qw(whois);
119       if ( $] < 5.006 ) {
120         eval "use Net::Whois::Raw 0.32 qw(whois)";
121         die $@ if $@;
122       }
123       use Text::CSV_XS;
124       use Spreadsheet::WriteExcel;
125       use Business::CreditCard 0.30; #for mask-aware cardtype()
126       use NetAddr::IP;
127       use String::Approx qw(amatch);
128       use Chart::LinesPoints;
129       use Chart::Mountain;
130       use Color::Scheme;
131       use HTML::Widgets::SelectLayers 0.07; #should go away in favor of
132                                             #selectlayers.html
133       use Locale::Country;
134       use Business::US::USPS::WebTools::AddressStandardization;
135       use FS;
136       use FS::UID qw( adminsuidsetup cgisuidsetup getotaker
137                       dbh datasrc driver_name
138                     );
139       use FS::Record qw(qsearch qsearchs fields dbdef str2time_sql);
140       use FS::Conf;
141       use FS::CGI qw(header menubar popurl rooturl table itable ntable idiot
142                      eidiot myexit http_header);
143       use FS::UI::Web qw(svc_url);
144       use FS::UI::Web::small_custview qw(small_custview);
145       use FS::UI::bytecount;
146       use FS::Msgcat qw(gettext geterror);
147       use FS::Misc qw( send_email send_fax states_hash counties state_label );
148       use FS::Report::Table::Monthly;
149       use FS::TicketSystem;
150
151       use FS::agent;
152       use FS::agent_type;
153       use FS::domain_record;
154       use FS::cust_bill;
155       use FS::cust_bill_pay;
156       use FS::cust_credit;
157       use FS::cust_credit_bill;
158       use FS::cust_main qw(smart_search);
159       use FS::cust_main_county;
160       use FS::cust_pay;
161       use FS::cust_pkg;
162       use FS::part_pkg_taxclass;
163       use FS::cust_pkg_reason;
164       use FS::cust_refund;
165       use FS::cust_credit_refund;
166       use FS::cust_pay_refund;
167       use FS::cust_svc;
168       use FS::nas;
169       use FS::part_bill_event;
170       use FS::part_event;
171       use FS::part_event_condition;
172       use FS::part_pkg;
173       use FS::part_referral;
174       use FS::part_svc;
175       use FS::part_svc_router;
176       use FS::part_virtual_field;
177       use FS::pay_batch;
178       use FS::pkg_svc;
179       use FS::port;
180       use FS::queue qw(joblisting);
181       use FS::raddb;
182       use FS::session;
183       use FS::svc_acct;
184       use FS::svc_acct_pop qw(popselector);
185       use FS::acct_rt_transaction;
186       use FS::svc_domain;
187       use FS::svc_forward;
188       use FS::svc_www;
189       use FS::router;
190       use FS::addr_block;
191       use FS::svc_broadband;
192       use FS::svc_external;
193       use FS::type_pkgs;
194       use FS::part_export;
195       use FS::part_export_option;
196       use FS::export_svc;
197       use FS::msgcat;
198       use FS::rate;
199       use FS::rate_region;
200       use FS::rate_prefix;
201       use FS::payment_gateway;
202       use FS::agent_payment_gateway;
203       use FS::XMLRPC;
204       use FS::payby;
205       use FS::cdr;
206       use FS::inventory_class;
207       use FS::inventory_item;
208       use FS::pkg_class;
209       use FS::access_user;
210       use FS::access_user_pref;
211       use FS::access_group;
212       use FS::access_usergroup;
213       use FS::access_groupagent;
214       use FS::access_right;
215       use FS::AccessRight;
216       use FS::svc_phone;
217       use FS::reason_type;
218       use FS::reason;
219       use FS::cust_main_note;
220       use FS::tax_class;
221       use FS::cust_tax_location;
222       use FS::part_pkg_taxproduct;
223       use FS::part_pkg_taxoverride;
224       use FS::part_pkg_taxrate;
225       use FS::tax_rate;
226
227       if ( %%%RT_ENABLED%%% ) {
228         eval '
229           use RT::Tickets;
230           use RT::Transactions;
231           use RT::Users;
232           use RT::CurrentUser;
233           use RT::Templates;
234           use RT::Queues;
235           use RT::ScripActions;
236           use RT::ScripConditions;
237           use RT::Scrips;
238           use RT::Groups;
239           use RT::GroupMembers;
240           use RT::CustomFields;
241           use RT::CustomFieldValues;
242           use RT::ObjectCustomFieldValues;
243
244           #blah.  manually updated from RT::Interface::Web::Handler
245           use RT::Interface::Web;
246           use MIME::Entity;
247           use Text::Wrapper;
248           use Time::ParseDate;
249           use Time::HiRes;
250           use HTML::Scrubber;
251
252           #blah.  not even in RT::Interface::Web::Handler, just in 
253           #html/NoAuth/css/dhandler and rt-test-dependencies.  ask for it here
254           #to throw a real error instead of just a mysterious unstyled RT
255           use CSS::Squish 0.06;
256
257           #slow, unreliable, segfaults and is optional
258           #see rt/html/Ticket/Elements/ShowTransactionAttachments
259           #use Text::Quoted;
260
261           #?#use File::Path qw( rmtree );
262           #?#use File::Glob qw( bsd_glob );
263           #?#use File::Spec::Unix;
264
265         ';
266         die $@ if $@;
267       }
268
269       *CGI::redirect = sub {
270         my $self = shift;
271         my $cookie = '';
272         if ( $_[0] eq '-cookie' ) { #this isn't actually used at the moment
273           (my $x, $cookie) = (shift, shift);
274           $HTML::Mason::r->err_headers_out->add( 'Set-cookie' => $cookie );
275         }
276         my $location = shift;
277
278         use vars qw($m);
279
280         # false laziness w/below
281         if ( defined(@DBIx::Profile::ISA) ) { #profiling redirect
282
283           my $page =
284             qq!<HTML><BODY>Redirect to <A HREF="$location">$location</A>!.
285             '<BR><BR><PRE>'.
286               ( UNIVERSAL::can(dbh, 'sprintProfile')
287                   ? encode_entities(dbh->sprintProfile())
288                   : 'DBIx::Profile missing sprintProfile method;'.
289                     'unpatched or too old?'                        ).
290             #"\n\n". &sprintAutoProfile().  '</PRE>'.
291             "\n\n".                         '</PRE>'.
292             '</BODY></HTML>';
293           dbh->{'private_profile'} = {};
294           return $page;
295
296         } else { #normal redirect
297
298           $m->redirect($location);
299           '';
300
301         }
302
303       };
304       
305       if ( $HTML::Mason::r->filename !~ /\/rt\/.*NoAuth/ ) { #not RT images/JS
306
307         $cgi = new CGI;
308         &cgisuidsetup($cgi);
309         #&cgisuidsetup($r);
310         $p = popurl(2);
311         $fsurl = rooturl();
312
313       } elsif ( $HTML::Mason::r->filename =~ /\/rt\/REST\/.*NoAuth/ ) {
314
315         #need to log somebody in for the mail gw
316
317         ##old installs w/fs_selfs or selfserv??
318         #&adminsuidsetup('fs_selfservice');
319
320         &adminsuidsetup('fs_queue');
321
322       }
323
324       sub include {
325         use vars qw($m);
326         $m->scomp(@_);
327       }
328
329       sub errorpage {
330         use vars qw($m);
331         $m->comp('/elements/errorpage.html', @_);
332       }
333
334       sub redirect {
335         my( $location ) = @_;
336         use vars qw($m);
337         $m->clear_buffer;
338         #false laziness w/above
339         if ( defined(@DBIx::Profile::ISA) ) { #profiling redirect
340
341           $m->print(
342             qq!<HTML><BODY>Redirect to <A HREF="$location">$location</A>!.
343             '<BR><BR><PRE>'.
344               ( UNIVERSAL::can(dbh, 'sprintProfile')
345                   ? encode_entities(dbh->sprintProfile())
346                   : 'DBIx::Profile missing sprintProfile method;'.
347                     'unpatched or too old?'                        ).
348             #"\n\n". &sprintAutoProfile().  '</PRE>'.
349             "\n\n".                         '</PRE>'.
350             '</BODY></HTML>'
351           );
352           dbh->{'private_profile'} = {};
353
354           #whew.  removing this is all that's needed to fix the annoying
355           #blank-page-instead-of-profiling-redirect-when-called-from-an-include
356           #bug triggered by mason 1.32
357           #my $rv = $m->abort(200);
358
359         } else { #normal redirect
360
361           $m->redirect($location);
362
363         }
364
365       }
366
367     } # end package HTML::Mason::Commands;
368
369     ###Module::Refresh->refresh;###
370
371     $r->content_type('text/html');
372     #eorar
373
374     my $headers = $r->headers_out;
375     $headers->{'Cache-control'} = 'no-cache';
376     #$r->no_cache(1);
377     $headers->{'Expires'} = '0';
378
379 #    $r->send_http_header;
380
381     if ( $r->filename =~ /\/rt\// ) { #RT
382
383       $ah->interp($rt_interp);
384
385       local $SIG{__WARN__};
386       local $SIG{__DIE__};
387
388       RT::Init();
389
390       # We don't need to handle non-text, non-xml items
391       return -1 if defined( $r->content_type )
392                 && $r->content_type !~ m!(^text/|\bxml\b)!io;
393
394     } else {
395
396       $ah->interp($fs_interp);
397
398     }
399
400     my %session;
401     my $status;
402     eval { $status = $ah->handle_request($r); };
403 #!!
404 #    if ( $@ ) {
405 #       $RT::Logger->crit($@);
406 #    }
407     warn $@ if $@;
408
409     undef %session;
410
411 #!!
412 #    if ($RT::Handle->TransactionDepth) {
413 #       $RT::Handle->ForceRollback;
414 #       $RT::Logger->crit(
415 #"Transaction not committed. Usually indicates a software fault. Data loss may have occurred"
416 #       );
417 #    }
418
419     $status;
420 }
421
422 1;