Jérôme Augé wrote:
After making your example work, I don't see any segfaults. Please try again with modperl-1.99_08 which was released a few days ago.Hi,I'm beginning with mod_perl (mod_perl-1.99_05 + Apache 2.0.40 from RedHat 8.0) and I want to write a module for rewriting the documents that passes through the Apache proxy. So, I looked at the Apache::AdBlocker (http://perl.apache.org/docs/tutorials/tips/mod_perl_tricks/mod_perl_tricks.html#A_Banner_Ad_Blocker) module and I'm facing some problems for writing the content of the documents back to the client. My main problem is that I get a SEGFAULT when calling the "$r->print()" or "$r->send_http_header()" method. I get the request, copy the headers from "headers_in", make my own request with LWP, copy the headers to "headers_out", then it SEGFAULT when writing the document ... Are this methods deprecated/not fully implemented ? what is the correct way to write data to the client ? The other problem is that if I use the "$r->push_handlers(PerlHandler => \&proxy_handler)" mechanism, my "proxy_handler()" function is never called, so I do the work directly into the handler sub, is this ok ? I attached my test module below (I register it with a "PerlTransHandler Apache::Plop" statement in httpd.conf)
I've attached Plop.pm that apparently works. Hope that this is what you wanted to accomplish. I've used the following config:
<Location /plop/>
SetHandler perl-script
PerlHeaderParserHandler Apache::Plop
</Location>
Now to your code:
1. You can't push_handlers when you are inside a response handler.
Use PerlHeaderParserHandler instead
2. $r->headers_in->do() expects a return value and will abort on 0; see the attached code
also it should be:
$request->header( $_[0] => $_[1] );
instead of:
$request->header( {$_[0]} => $_[1] );
have you looked at error_log? You'd have seen that error reported.
3. This is not good:
my $request = HTTP::Request->new( $r->method, $r->uri);
since you don't the whole url. Use this instead:
my $request = HTTP::Request->new( $r->method, $r->construct_url);
this requires 'use Apache::URI'
4. Finally I've used a special header: (which can be anything)
$request->header( GetReal => 1 );
to know that now I'm inside the real request.
Hope that this helps.
Also you might want to use a sub-request rather than a heavy weighted LWP to accomplish what you do.
__________________________________________________________________
Stas Bekman JAm_pH ------> Just Another mod_perl Hacker
http://stason.org/ mod_perl Guide ---> http://perl.apache.org
mailto:[EMAIL PROTECTED] http://use.perl.org http://apacheweek.com
http://modperlbook.org http://apache.org http://ticketmaster.com
package Apache::Plop;
use strict;
use Apache::RequestRec;
use Apache::RequestIO;
use Apache::RequestUtil;
use Apache::Const;
use Apache::ServerUtil;
use Apache::Response;
use Apache::URI;
use APR::Table;
use LWP::UserAgent;
my $ua = LWP::UserAgent->new();
sub handler {
my $r = shift;
if( $r->proxyreq ) {
return Apache::DECLINED;
}
print STDERR "Good, this is a proxyreq ...\n";
$r->handler("perl-script"); #ok, let's do it
$r->push_handlers(PerlResponseHandler => \&proxy_handler);
return Apache::OK;
}
sub proxy_handler {
my $r = shift;
if( $r->method ne "GET" ) {
return Apache::DECLINED;
}
print STDERR "Good, this is a GET method ...\n";
if ( ($r->headers_in->get('GetReal')||0) == 1) {
$r->content_type('text/plain');
print "hey";
return Apache::OK;
}
# prepare the "real" request
my $request = HTTP::Request->new( $r->method, $r->construct_url);
# copy headers from client request
my %headers_in;
print STDERR "-- client headers --\n";
$r->headers_in()->do(
sub {
warn "$_[0]: $_[1]\n";
$headers_in{ $_[0] } = $_[1];
$request->header( $_[0] => $_[1] );
return 1;
}
);
print STDERR "-- end --\n";
# make the "real" request myself
$ua->agent( $headers_in{ 'User-Agent' } );
$request->header( GetReal => 1 );
warn $request->as_string;
my $response = $ua->request( $request );
if ( ! $response->is_success() ) {
print STDERR "== ERROR ==\n";
return Apache::DECLINED;
}
print STDERR "-- server headers --\n";
my %headers_out;
$response->headers()->scan(
sub {
print STDERR "$_[0]: $_[1]\n";
$headers_out{$_[0]} = $_[1];
}
);
print STDERR "-- end --\n";
# simply override the content
my $content = $response->content;
$content = "<html><body>plop</body></html>";
# adjust the headers for the new content
$headers_out{ 'Content-length' } = length( $content );
$headers_out{ 'Content-type' } = 'text/html';
# copy the modified response headers back to Apache
foreach (keys %headers_out) {
$r->headers_out->{$_} = $headers_out{$_};
}
$r->content_type( $headers_out{ 'Content-type' } );
print STDERR "-- send/print --\n";
$r->send_http_header();
$r->print( $content );
print STDERR "-- end --\n";
return Apache::OK;
}
1;
