aboutsummaryrefslogtreecommitdiff
path: root/lib
diff options
context:
space:
mode:
authorroot <root@rbsd.ankarstrom.se>2021-04-26 23:36:30 +0000
committerroot <root@rbsd.ankarstrom.se>2021-04-26 23:36:30 +0000
commit356ace1a5f618d00f145b640acb989dd0197e8d6 (patch)
tree0d04c610d49a213d4be1403bb81954e7ef7e6677 /lib
parent07b6aaa59257778fba1f805ee4051baf1917980b (diff)
downloadApache-Inject-356ace1a5f618d00f145b640acb989dd0197e8d6.tar.gz
Use filter instead of handler
This makes it work for HTML content that has already been processed, e.g., by PHP. This change was delightfully easy to make.
Diffstat (limited to 'lib')
-rw-r--r--lib/Apache/Inject.pm32
-rw-r--r--lib/Apache/Inject/Filter.pm (renamed from lib/Apache/Inject/Handler.pm)45
2 files changed, 40 insertions, 37 deletions
diff --git a/lib/Apache/Inject.pm b/lib/Apache/Inject.pm
index 0670e69..bf50728 100644
--- a/lib/Apache/Inject.pm
+++ b/lib/Apache/Inject.pm
@@ -31,8 +31,7 @@ sub Inject {
# Add relevant directives to current configuration
$parms->add_config([
- 'SetHandler perl-script',
- 'PerlResponseHandler Apache::Inject::Handler',
+ 'PerlOutputFilterHandler Apache::Inject::Filter',
qq{PerlSetVar InjectHead "$head"},
qq{PerlSetVar InjectFoot "$foot"},
]);
@@ -164,25 +163,26 @@ to a hyphen.
=head1 OPERATION
Behind the scenes, the Inject directive works as an alias for
-PerlResponseHandler and PerlSetVar. For example, C<Inject head.html
-foot.html> results in the following configuration being added:
+PerlOutputFilterHandler and PerlSetVar. For example, C<Inject
+head.html foot.html> results in the following configuration being
+added:
- PerlResponseHandler Apache::Inject::Handler
+ PerlOutputFilterHandler Apache::Inject::Filter
PerlSetVar InjectHeader head.html
PerlSetVar InjectFooter foot.html
-This results in Apache::Inject::Handler being registered as a handler
-for requests to the current directory or location.
+This results in Apache::Inject::Filter being registered as an output
+filter for requests to the current directory or location.
-Apache::Inject::Handler accepts all requests to files where the
-content type is C<text/html>. It reads the contents of the requested
-file, as well as the contents of the C<InjectHeader> and C<InjectFooter>
-files, concatenates them intelligently and prints their combined
-contents.
+Apache::Inject::Filter accepts all requests where the content type
+is C<text/html>. It receives the contents of the original page
+from Apache and, in addition, reads the contents of the C<InjectHeader>
+and C<InjectFooter> files. It then concatenates all of these
+intelligently and forwards their combined contents.
=head1 CAVEATS
-Apache::Inject::Handler uses regular expressions to determine the
+Apache::Inject::Filter uses regular expressions to determine the
proper location of the injected header. It supports all valid HTML.
However, it does not take into account that embedded CSS and
JavaScript code can contain strings that look like valid opening
@@ -194,11 +194,11 @@ without the module; it is only the tests that require it.
=head1 DIAGNOSTICS
-Apache::Inject and Apache::Inject::Handler log all errors and
+Apache::Inject and Apache::Inject::Filter log all errors and
warnings to the Apache log file. Below is a list of all issued
errors and warnings.
-Note that whenever Apache::Inject::Handler issues an error or a
+Note that whenever Apache::Inject::Filter issues an error or a
warning, this means that it also declines the request, letting
Apache handle it as it would if the Inject directive were not used.
@@ -227,7 +227,7 @@ exist.
=item Warning: Declining request due to empty document root
-This warning is issued if Apache::Inject::Handler for some reason
+This warning is issued if Apache::Inject::Filter for some reason
cannot retrieve the current document root from Apache.
=back
diff --git a/lib/Apache/Inject/Handler.pm b/lib/Apache/Inject/Filter.pm
index ae81fbb..f87cdc1 100644
--- a/lib/Apache/Inject/Handler.pm
+++ b/lib/Apache/Inject/Filter.pm
@@ -1,9 +1,10 @@
-package Apache::Inject::Handler;
+package Apache::Inject::Filter;
use 5.010000;
use strict;
use mod_perl2;
+use base qw/Apache2::Filter/;
use Apache2::Const qw/OK DECLINED/;
use Apache2::Log ();
use Apache2::RequestRec ();
@@ -37,53 +38,55 @@ my $doc = qr{
\z
}xmsi;
-sub handler {
- my $r = shift;
+sub handler : FilterRequestHandler {
+ my $f = shift;
- return DECLINED if not $r->content_type eq 'text/html';
+ return DECLINED if not $f->r->content_type;
+ return DECLINED if not $f->r->content_type =~ m{^text/html($|;.*)};
- my $content = ${$r->slurp_filename};
- return DECLINED if not $content =~ /$doc/;
-
- if (not $r->document_root) {
- $r->warn('Inject: Declining request due to empty document root');
+ if (not $f->r->document_root) {
+ $f->r->warn('Inject: Declining due to empty document root');
return DECLINED;
}
- print $+{head} if $+{head};
- inject($r, "InjectHead");
- print $+{body} if $+{body};
- inject($r, "InjectFoot");
- print $+{rest} if $+{rest};
+ my ($buf, $content);
+ $content .= $buf while $f->read($buf);
+ return DECLINED if not $content =~ /$doc/;
+
+ $f->print($+{head}) if $+{head};
+ inject($f, "InjectHead");
+ $f->print($+{body}) if $+{body};
+ inject($f, "InjectFoot");
+ $f->print($+{rest}) if $+{rest};
return OK;
}
sub inject {
- my ($r, $var) = @_;
+ my ($f, $var) = @_;
# Retrieve value implicitly set by Inject directive
- return if not (my $val = $r->dir_config($var));
+ return if not (my $val = $f->r->dir_config($var));
return if $val eq '-'; # special value signifying absence of argument
# Validate path
if ($val =~ m{^/}) {
- $r->log_error("Inject: $var should not begin with slash, as it is already always relative to document root");
+ $f->r->log_error("Inject: $var should not begin with slash, as it is already always relative to document root");
}
if ($val =~ m{^../|/../|/..$}) {
- $r->log_error("Inject: $var cannot extend past document root");
+ $f->r->log_error("Inject: $var cannot extend past document root");
return;
}
# note: document root has been confirmed not to be empty
- my $root = $r->document_root;
+ my $root = $f->r->document_root;
# Read contents of specified file
open my $fh, '<', "$root/$val" or do {
- $r->log_error("Inject: $var $root/$val does not exist");
+ $f->r->log_error("Inject: $var $root/$val does not exist");
return;
};
- print for <$fh>;
+ $f->print($_) for <$fh>;
close $fh;
}