package HTML::Mason;
use strict;
-use vars qw($r);
-use HTML::Mason 1.27; #http://www.masonhq.com/?ApacheModPerl2Redirect
-use HTML::Mason::Interp;
-use HTML::Mason::Compiler::ToObject;
+use warnings;
+use FS::Mason qw( mason_interps );
+use FS::Trace;
+use FS::access_user_log;
+use FS::Conf;
+
+$FS::Conf::conf_cache_enabled = 1; # enable FS::Conf caching for performance
+
+# Preload to share in mod_perl parent for performance
+use FS::UID qw(load_schema);
+load_schema();
+use FS::Record qw(fk_methods_init);
+fk_methods_init;
+
+if ( %%%RT_ENABLED%%% ) {
+
+ require RT;
+
+ $> = scalar(getpwnam('freeside'));
+
+ RT::LoadConfig();
+ RT::Init();
+
+ # disconnect DB before fork:
+ # (avoid 'prepared statement "dbdpg_p\d+_\d+" already exists' errors?)
+ $RT::Handle->dbh(undef);
+ undef $RT::Handle;
+
+ $> = $<;
+}
+
+#use vars qw($r);
# Bring in ApacheHandler, necessary for mod_perl integration.
# Uncomment the second line (and comment the first) to use
###use Module::Refresh;###
-# List of modules that you want to use from components (see Admin
-# manual for details)
-#{ package HTML::Mason::Commands;
-# use CGI;
-#}
-
-if ( %%%RT_ENABLED%%% ) {
- eval '
- use lib ( "/opt/rt3/local/lib", "/opt/rt3/lib" );
- use RT;
- use vars qw($Nobody $SystemUser);
- RT::LoadConfig();
- ';
- die $@ if $@;
-}
-
# Create Mason objects
-my %interp = (
- request_class => 'HTML::Mason::Request::ApacheHandler',
- data_dir => '%%%MASONDATA%%%',
- error_mode => 'output',
- error_format => 'html',
- ignore_warnings_expr => '.',
- comp_root => [
- [ 'freeside' => '%%%FREESIDE_DOCUMENT_ROOT%%%' ],
- [ 'rt' => '%%%FREESIDE_DOCUMENT_ROOT%%%/rt' ],
- ],
-);
-
-my $fs_interp = new HTML::Mason::Interp (
- %interp,
- escape_flags => { 'js_string' => sub {
- #${$_[0]} =~ s/(['\\\n])/'\\'.($1 eq "\n" ? 'n' : $1)/ge;
- ${$_[0]} =~ s/(['\\])/\\$1/g;
- ${$_[0]} =~ s/\n/\\n/g;
- ${$_[0]} = "'". ${$_[0]}. "'";
- }
- },
-);
-
-my $rt_interp = new HTML::Mason::Interp (
- %interp,
- escape_flags => { 'h' => \&RT::Interface::Web::EscapeUTF8 },
- compiler => HTML::Mason::Compiler::ToObject->new(
- default_escape_flags => 'h',
- allow_globals => [qw(%session)],
- ),
-);
+my( $fs_interp, $rt_interp ) = mason_interps('apache');
my $ah = new HTML::Mason::ApacheHandler (
- interp => $fs_interp,
- args_method => 'CGI', #(and FS too)
+ interp => $fs_interp,
+ request_class => 'FS::Mason::Request',
+ args_method => 'CGI', #(and FS too)
);
# Activate the following if running httpd as root (the normal case).
#
#chown (Apache->server->uid, Apache->server->gid, $interp->files_written);
+my $protect_fds;
+
sub handler
{
- ($r) = @_;
+ #($r) = @_;
+ my $r = shift;
+
+ my $start_time = time;
+
+ FS::Trace->log('protecting fds');
+
+ #from rt/bin/webmux.pl(.in)
+ if ( !$protect_fds && $ENV{'MOD_PERL'} && exists $ENV{'MOD_PERL_API_VERSION'}
+ && $ENV{'MOD_PERL_API_VERSION'} >= 2
+ ) {
+ # under mod_perl2, STDIN and STDOUT get closed and re-opened,
+ # however they are not on FD 0 and 1. In this case, the next
+ # socket that gets opened will occupy one of these FDs, and make
+ # all system() and open "|-" calls dangerous; for example, the
+ # DBI handle can get this FD, which later system() calls will
+ # close by putting garbage into the socket.
+ $protect_fds = [];
+ push @{$protect_fds}, IO::Handle->new_from_fd(0, "r")
+ if fileno(STDIN) != 0;
+ push @{$protect_fds}, IO::Handle->new_from_fd(1, "w")
+ if fileno(STDOUT) != 1;
+ }
# If you plan to intermix images in the same directory as
# components, activate the following to prevent Mason from
#
#return -1 if $r->content_type && $r->content_type !~ m|^text/|i;
- #rar
- { package HTML::Mason::Commands;
- use strict;
- use vars qw( $cgi $p $fsurl);
- use vars qw( %session );
- use CGI 2.47 qw(-private_tempfiles);
- #use CGI::Carp qw(fatalsToBrowser);
- use CGI::Cookie;
- use List::Util qw( max min );
- use Data::Dumper;
- use Date::Format;
- use Date::Parse;
- use Time::Local;
- use Time::Duration;
- use DateTime;
- use DateTime::Format::Strptime;
- use Lingua::EN::Inflect qw(PL);
- use Tie::IxHash;
- use URI::Escape;
- use HTML::Entities;
- use JSON;
- use MIME::Base64;
- use IO::Handle;
- use IO::File;
- use IO::Scalar;
- #not actually using this yet anyway...# use IPC::Run3 0.036;
- use Net::Whois::Raw qw(whois);
- if ( $] < 5.006 ) {
- eval "use Net::Whois::Raw 0.32 qw(whois)";
- die $@ if $@;
- }
- use Text::CSV_XS;
- use Spreadsheet::WriteExcel;
- use Business::CreditCard 0.30; #for mask-aware cardtype()
- use NetAddr::IP;
- use String::Approx qw(amatch);
- use Chart::LinesPoints;
- use Chart::Mountain;
- use Color::Scheme;
- use HTML::Widgets::SelectLayers 0.07;
- use Locale::Country;
- use FS;
- use FS::UID qw(cgisuidsetup dbh getotaker datasrc driver_name);
- use FS::Record qw(qsearch qsearchs fields dbdef str2time_sql);
- use FS::Conf;
- use FS::CGI qw(header menubar popurl rooturl table itable ntable idiot
- eidiot small_custview myexit http_header);
- use FS::UI::Web qw(svc_url);
- use FS::UI::bytecount;
- use FS::Msgcat qw(gettext geterror);
- use FS::Misc qw( send_email send_fax states_hash counties state_label );
- use FS::Report::Table::Monthly;
- use FS::TicketSystem;
-
- use FS::agent;
- use FS::agent_type;
- use FS::domain_record;
- use FS::cust_bill;
- use FS::cust_bill_pay;
- use FS::cust_credit;
- use FS::cust_credit_bill;
- use FS::cust_main qw(smart_search);
- use FS::cust_main_county;
- use FS::cust_pay;
- use FS::cust_pkg;
- use FS::cust_pkg_reason;
- use FS::cust_refund;
- use FS::cust_svc;
- use FS::nas;
- use FS::part_bill_event;
- use FS::part_event;
- use FS::part_event_condition;
- use FS::part_pkg;
- use FS::part_referral;
- use FS::part_svc;
- use FS::part_svc_router;
- use FS::part_virtual_field;
- use FS::pay_batch;
- use FS::pkg_svc;
- use FS::port;
- use FS::queue qw(joblisting);
- use FS::raddb;
- use FS::session;
- use FS::svc_acct;
- use FS::svc_acct_pop qw(popselector);
- use FS::acct_rt_transaction;
- use FS::svc_domain;
- use FS::svc_forward;
- use FS::svc_www;
- use FS::router;
- use FS::addr_block;
- use FS::svc_broadband;
- use FS::svc_external;
- use FS::type_pkgs;
- use FS::part_export;
- use FS::part_export_option;
- use FS::export_svc;
- use FS::msgcat;
- use FS::rate;
- use FS::rate_region;
- use FS::rate_prefix;
- use FS::payment_gateway;
- use FS::agent_payment_gateway;
- use FS::XMLRPC;
- use FS::payby;
- use FS::cdr;
- use FS::inventory_class;
- use FS::inventory_item;
- use FS::pkg_class;
- use FS::access_user;
- use FS::access_user_pref;
- use FS::access_group;
- use FS::access_usergroup;
- use FS::access_groupagent;
- use FS::access_right;
- use FS::AccessRight;
- use FS::svc_phone;
- use FS::reason_type;
- use FS::reason;
- use FS::cust_main_note;
-
- if ( %%%RT_ENABLED%%% ) {
- eval '
- use RT::Tickets;
- use RT::Transactions;
- use RT::Users;
- use RT::CurrentUser;
- use RT::Templates;
- use RT::Queues;
- use RT::ScripActions;
- use RT::ScripConditions;
- use RT::Scrips;
- use RT::Groups;
- use RT::GroupMembers;
- use RT::CustomFields;
- use RT::CustomFieldValues;
- use RT::ObjectCustomFieldValues;
-
- use RT::Interface::Web;
- use MIME::Entity;
- use Text::Wrapper;
- use Time::ParseDate;
- use HTML::Scrubber;
- use Text::Quoted;
- ';
- die $@ if $@;
- }
-
- *CGI::redirect = sub {
- my $self = shift;
- my $cookie = '';
- if ( $_[0] eq '-cookie' ) { #this isn't actually used at the moment
- (my $x, $cookie) = (shift, shift);
- $HTML::Mason::r->err_headers_out->add( 'Set-cookie' => $cookie );
- }
- my $location = shift;
-
- use vars qw($m);
-
- # false laziness w/below
- if ( defined(@DBIx::Profile::ISA) ) { #profiling redirect
-
- my $page =
- qq!<HTML><BODY>Redirect to <A HREF="$location">$location</A>!.
- '<BR><BR><PRE>'.
- ( UNIVERSAL::can(dbh, 'sprintProfile')
- ? encode_entities(dbh->sprintProfile())
- : 'DBIx::Profile missing sprintProfile method;'.
- 'unpatched or too old?' ).
- #"\n\n". &sprintAutoProfile(). '</PRE>'.
- "\n\n". '</PRE>'.
- '</BODY></HTML>';
- dbh->{'private_profile'} = {};
- return $page;
-
- } else { #normal redirect
-
- $m->redirect($location);
- '';
-
- }
-
- };
-
- unless ( $HTML::Mason::r->filename =~ /\/rt\/.*NoAuth/ ) { #RT
- $cgi = new CGI;
- &cgisuidsetup($cgi);
- #&cgisuidsetup($r);
- $p = popurl(2);
- $fsurl = rooturl();
- }
-
- sub include {
- use vars qw($m);
- $m->scomp(@_);
- }
-
- sub errorpage {
- use vars qw($m);
- $m->comp('/elements/errorpage.html', @_);
- }
-
- sub redirect {
- my( $location ) = @_;
- use vars qw($m);
- $m->clear_buffer;
- #false laziness w/above
- if ( defined(@DBIx::Profile::ISA) ) { #profiling redirect
-
- $m->print(
- qq!<HTML><BODY>Redirect to <A HREF="$location">$location</A>!.
- '<BR><BR><PRE>'.
- ( UNIVERSAL::can(dbh, 'sprintProfile')
- ? encode_entities(dbh->sprintProfile())
- : 'DBIx::Profile missing sprintProfile method;'.
- 'unpatched or too old?' ).
- #"\n\n". &sprintAutoProfile(). '</PRE>'.
- "\n\n". '</PRE>'.
- '</BODY></HTML>'
- );
- dbh->{'private_profile'} = {};
-
- #whew. removing this is all that's needed to fix the annoying
- #blank-page-instead-of-profiling-redirect-when-called-from-an-include
- #bug triggered by mason 1.32
- #my $rv = $m->abort(200);
-
- } else { #normal redirect
-
- $m->redirect($location);
-
- }
-
- }
-
- } # end package HTML::Mason::Commands;
-
###Module::Refresh->refresh;###
- $r->content_type('text/html');
+ FS::Trace->log('setting content_type / headers');
+
+ $r->content_type('text/html; charset=utf-8');
+ #$r->content_type('text/html; charset=iso-8859-1');
#eorar
my $headers = $r->headers_out;
if ( $r->filename =~ /\/rt\// ) { #RT
- $ah->interp($rt_interp);
+ FS::Trace->log('handling RT file');
+
+ # We don't need to handle non-text, non-xml items
+ return -1 if defined( $r->content_type )
+ && $r->content_type !~ m!(^text/|\bxml\b)!io;
local $SIG{__WARN__};
local $SIG{__DIE__};
- RT::Init();
+ FS::Trace->log('initializing RT');
+ my_rt_init();
- # We don't need to handle non-text, non-xml items
- return -1 if defined( $r->content_type )
- && $r->content_type !~ m!(^text/|\bxml\b)!io;
+ FS::Trace->log('setting RT interpreter');
+ $ah->interp($rt_interp);
} else {
+ FS::Trace->log('handling Freeside file');
+
+ local $SIG{__WARN__};
+ local $SIG{__DIE__};
+
+ FS::Trace->log('initializing RT');
+ my_rt_init();
+
+ #we don't want the RT error handlers under FS
+ {
+ no warnings 'uninitialized';
+ undef($SIG{__WARN__}) if defined($SIG{__WARN__});
+ undef($SIG{__DIE__}) if defined($SIG{__DIE__} );
+ }
+
+ FS::Trace->log('setting Freeside interpreter');
$ah->interp($fs_interp);
}
+ FS::Trace->log('handling request');
my %session;
my $status;
eval { $status = $ah->handle_request($r); };
# );
# }
+ FS::access_user_log->insert_new_path( $r->filename, time-$start_time );
+
+ FS::Trace->log('done');
+
+ FS::Trace->dumpfile( "%%%FREESIDE_EXPORT%%%/profile/$$.".time,
+ FS::Trace->total. ' '. $r->filename
+ )
+ if FS::Trace->total > 5; #10?
+
+ FS::Trace->reset;
+
$status;
}
+sub my_rt_init {
+ return unless $RT::VERSION;
+ RT::ConnectToDatabase();
+ RT::InitSignalHandlers();
+}
+
1;