1022 lines
34 KiB
Prolog
Executable File
1022 lines
34 KiB
Prolog
Executable File
#!/usr/bin/perl
|
|
#
|
|
# <LJDEP>
|
|
# lib: HTML::TokeParser, cgi-bin/ljconfig.pl, cgi-bin/ljlib.pl
|
|
# link: htdocs/userinfo.bml, htdocs/users
|
|
# </LJDEP>
|
|
|
|
require "$ENV{'LJHOME'}/cgi-bin/ljconfig.pl";
|
|
|
|
use strict;
|
|
use HTML::TokeParser ();
|
|
use URI ();
|
|
|
|
# LJ::CleanHTML::clean(\$u->{'bio'}, {
|
|
# 'wordlength' => 100, # maximum length of an unbroken "word"
|
|
# 'addbreaks' => 1, # insert <br/> after newlines where appropriate
|
|
# 'tablecheck' => 1, # make sure they aren't closing </td> that weren't opened.
|
|
# 'eat' => [qw(head title style layer iframe)],
|
|
# 'mode' => 'allow',
|
|
# 'deny' => [qw(marquee)],
|
|
# 'remove' => [qw()],
|
|
# 'maximgwidth' => 100,
|
|
# 'maximgheight' => 100,
|
|
# 'keepcomments' => 1,
|
|
# 'cuturl' => 'http://www.domain.com/full_item_view.ext',
|
|
# 'ljcut_disable' => 1, # stops the cleaner from using the lj-cut tag
|
|
# 'cleancss' => 1,
|
|
# 'extractlinks' => 1, # remove a hrefs; implies noautolinks
|
|
# 'noautolinks' => 1, # do not auto linkify
|
|
# 'extractimages' => 1, # placeholder images
|
|
# });
|
|
|
|
package LJ::CleanHTML;
|
|
|
|
sub helper_preload
|
|
{
|
|
my $p = HTML::TokeParser->new("");
|
|
eval {$p->DESTROY(); };
|
|
}
|
|
|
|
|
|
# this treats normal characters and &entities; as single characters
|
|
# also treats UTF-8 chars as single characters if $LJ::UNICODE
|
|
my $onechar;
|
|
{
|
|
my $utf_longchar = '[\xc2-\xdf][\x80-\xbf]|\xe0[\xa0-\xbf][\x80-\xbf]|[\xe1-\xef][\x80-\xbf][\x80-\xbf]|\xf0[\x90-\xbf][\x80-\xbf][\x80-\xbf]|[\xf1-\xf7][\x80-\xbf][\x80-\xbf][\x80-\xbf]';
|
|
my $match;
|
|
if (not $LJ::UNICODE) {
|
|
$match = '[^&\s]|(&\#?\w{1,7};)';
|
|
} else {
|
|
$match = $utf_longchar . '|[^&\s\x80-\xff]|(?:&\#?\w{1,7};)';
|
|
}
|
|
$onechar = qr/$match/o;
|
|
}
|
|
|
|
# Some browsers, such as Internet Explorer, have decided to alllow
|
|
# certain HTML tags to be an alias of another. This has manifested
|
|
# itself into a problem, as these aliases act in the browser in the
|
|
# same manner as the original tag, but are not treated the same by
|
|
# the HTML cleaner.
|
|
# 'alias' => 'real'
|
|
my %tag_substitute = (
|
|
'image' => 'img',
|
|
);
|
|
|
|
# <LJFUNC>
|
|
# name: LJ::CleanHTML::clean
|
|
# class: text
|
|
# des: Multifaceted HTML parse function
|
|
# info:
|
|
# args: data, opts
|
|
# des-data: A reference to html to parse to output, or HTML if modified in-place.
|
|
# des-opts: An hash of options to pass to the parser.
|
|
# returns: Nothing.
|
|
# </LJFUNC>
|
|
|
|
sub clean
|
|
{
|
|
my $data = shift;
|
|
my $opts = shift;
|
|
my $newdata;
|
|
|
|
# remove the auth portion of any see_request.bml links
|
|
$$data =~ s/(see_request\.bml.+?)auth=\w+/$1/ig;
|
|
|
|
my $p = HTML::TokeParser->new($data);
|
|
|
|
my $wordlength = $opts->{'wordlength'};
|
|
my $addbreaks = $opts->{'addbreaks'};
|
|
my $keepcomments = $opts->{'keepcomments'};
|
|
my $mode = $opts->{'mode'};
|
|
my $cut = $opts->{'cuturl'} || $opts->{'cutpreview'};
|
|
my $ljcut_disable = $opts->{'ljcut_disable'};
|
|
my $s1var = $opts->{'s1var'};
|
|
my $extractlinks = 0 || $opts->{'extractlinks'};
|
|
my $noautolinks = $extractlinks || $opts->{'noautolinks'};
|
|
|
|
my @canonical_urls; # extracted links
|
|
|
|
my %action = ();
|
|
my %remove = ();
|
|
if (ref $opts->{'eat'} eq "ARRAY") {
|
|
foreach (@{$opts->{'eat'}}) { $action{$_} = "eat"; }
|
|
}
|
|
if (ref $opts->{'allow'} eq "ARRAY") {
|
|
foreach (@{$opts->{'allow'}}) { $action{$_} = "allow"; }
|
|
}
|
|
if (ref $opts->{'deny'} eq "ARRAY") {
|
|
foreach (@{$opts->{'deny'}}) { $action{$_} = "deny"; }
|
|
}
|
|
if (ref $opts->{'remove'} eq "ARRAY") {
|
|
foreach (@{$opts->{'remove'}}) { $action{$_} = "deny"; $remove{$_} = 1; }
|
|
}
|
|
|
|
$action{'script'} = "eat";
|
|
|
|
my @attrstrip = qw();
|
|
if ($opts->{'cleancss'}) {
|
|
push @attrstrip, 'id';
|
|
}
|
|
|
|
if (ref $opts->{'attrstrip'} eq "ARRAY") {
|
|
foreach (@{$opts->{'attrstrip'}}) { push @attrstrip, $_; }
|
|
}
|
|
|
|
my %opencount = ();
|
|
my @tablescope = ();
|
|
|
|
my $cutcount = 0;
|
|
|
|
my $total_fail = sub {
|
|
my $tag = LJ::ehtml(@_);
|
|
$$data = LJ::ehtml($$data);
|
|
$$data =~ s/\r?\n/<br \/>/g if $addbreaks;
|
|
$$data = "[<b>Error:</b> Irreparable invalid markup ('<$tag>') in entry. ".
|
|
"Owner must fix manually. Raw contents below.]<br /><br />" .
|
|
'<div style="width: 95%; overflow: auto">' .
|
|
$$data .
|
|
'</div>';
|
|
return undef;
|
|
};
|
|
|
|
TOKEN:
|
|
while (my $token = $p->get_token)
|
|
{
|
|
my $type = $token->[0];
|
|
|
|
# See if this tag should be treated as an alias
|
|
|
|
$token->[1] = $tag_substitute{$token->[1]} if defined $tag_substitute{$token->[1]} &&
|
|
($type eq 'S' || $type eq 'E');
|
|
|
|
if ($type eq "S") # start tag
|
|
{
|
|
my $tag = $token->[1];
|
|
|
|
# check for exploit using TIME namespace to set data and introduce javascript that
|
|
# would affect IE users.
|
|
if ($tag =~ m/:set$/ && $token->[2]->{attributename} =~ /innerHTML/i) {
|
|
next;
|
|
}
|
|
|
|
# do some quick checking to see if this is an email address/URL, and if so, just
|
|
# escape it and ignore it
|
|
if ($tag =~ m!(?:\@|://)!) {
|
|
$newdata .= LJ::ehtml("<$tag>");
|
|
next;
|
|
}
|
|
|
|
my $slashclose = 0; # If set to 1, use XML-style empty tag marker
|
|
# for tags like <name/>, pretend it's <name> and reinsert the slash later
|
|
$slashclose = 1 if ($tag =~ s!/$!!);
|
|
|
|
return $total_fail->($tag) unless $tag =~ /^\w([\w\-:_]*\w)?$/;
|
|
|
|
# for incorrect tags like <name/attrib=val> (note the lack of a space)
|
|
# delete everything after 'name' to prevent a security loophole which happens
|
|
# because IE understands them.
|
|
$tag =~ s!/.+$!!;
|
|
|
|
if ($action{$tag} eq "eat") {
|
|
$p->unget_token($token);
|
|
$p->get_tag("/$tag");
|
|
}
|
|
elsif ($tag eq "lj-cut" && !$ljcut_disable)
|
|
{
|
|
my $attr = $token->[2];
|
|
$cutcount++;
|
|
if ($cut) {
|
|
my $text = "Read more...";
|
|
if ($attr->{'text'}) {
|
|
$text = $attr->{'text'};
|
|
if ($text =~ /[^\x01-\x7f]/) {
|
|
$text = pack('C*', unpack('C*', $text));
|
|
}
|
|
$text =~ s/</</g;
|
|
$text =~ s/>/>/g;
|
|
}
|
|
my $url = LJ::ehtml($cut);
|
|
$newdata .= "<b>( <a href=\"$url#cutid$cutcount\">$text</a> )</b>";
|
|
$p->get_tag("/lj-cut") unless $opts->{'cutpreview'}
|
|
} else {
|
|
$newdata .= "<a name=\"cutid$cutcount\"></a>";
|
|
next;
|
|
}
|
|
}
|
|
elsif ($tag eq "lj")
|
|
{
|
|
my $attr = $token->[2];
|
|
|
|
# keep <lj comm> working for backwards compatibility, but pretend
|
|
# it was <lj user> so we don't have to account for it below.
|
|
my $user = $attr->{'user'} = exists $attr->{'user'} ? $attr->{'user'} :
|
|
exists $attr->{'comm'} ? $attr->{'comm'} : undef;
|
|
|
|
if (length $user) {
|
|
$user = LJ::canonical_username($user);
|
|
if ($s1var) {
|
|
$newdata .= "%%ljuser:$1%%" if $attr->{'user'} =~ /^\%\%([\w\-\']+)\%\%$/;
|
|
} elsif (length $user) {
|
|
if ($opts->{'textonly'}) {
|
|
$newdata .= $user;
|
|
} else {
|
|
$newdata .= LJ::ljuser($user);
|
|
}
|
|
} else {
|
|
$newdata .= "<b>[Bad username in LJ tag]</b>";
|
|
}
|
|
} else {
|
|
$newdata .= "<b>[Unknown LJ tag]</b>";
|
|
}
|
|
}
|
|
elsif ($tag eq "lj-raw")
|
|
{
|
|
# Strip it out, but still register it as being open
|
|
$opencount{$tag}++;
|
|
}
|
|
else
|
|
{
|
|
my $alt_output = 0;
|
|
|
|
my $hash = $token->[2];
|
|
my $attrs = $token->[3]; # attribute names, in original order
|
|
|
|
$slashclose = 1 if delete $hash->{'/'};
|
|
|
|
foreach (@attrstrip) {
|
|
delete $hash->{$_};
|
|
}
|
|
|
|
if ($tag eq "form") {
|
|
my $action = lc($hash->{'action'});
|
|
my $deny = 0;
|
|
if ($action =~ m!^https?://?([^/]+)!) {
|
|
my $host = $1;
|
|
$deny = 1 if
|
|
$host =~ /[%\@\s]/ ||
|
|
$LJ::FORM_DOMAIN_BANNED{$host};
|
|
} else {
|
|
$deny = 1;
|
|
}
|
|
delete $hash->{'action'} if $deny;
|
|
}
|
|
|
|
foreach my $attr (keys %$hash)
|
|
{
|
|
delete $hash->{$attr} if $attr =~ /^(?:on|dynsrc|data)/;
|
|
if ($attr =~ /(?:^=)|[\x0b\x0d]/) {
|
|
# Cleaner attack: <p ='>' onmouseover="javascript:alert(document/**/.cookie)" >
|
|
# is returned by HTML::Parser as P_tag("='" => "='") Text( onmouseover...)
|
|
# which leads to reconstruction of valid HTML. Clever!
|
|
# detect this, and fail.
|
|
return $total_fail->("$tag $attr");
|
|
}
|
|
|
|
# ignore attributes that do not fit this strict scheme
|
|
return $total_fail->("$tag " . (%$hash > 1 ? "[...] " : "") . "$attr")
|
|
unless $attr =~ /^[\w_:-]+$/;
|
|
|
|
$hash->{$attr} =~ s/[\t\n]//g;
|
|
# IE sucks:
|
|
if ($hash->{$attr} =~ /(j\s*a\s*v\s*a\s*s\s*c\s*r\s*i\s*p\s*t|
|
|
v\s*b\s*s\s*c\s*r\s*i\s*p\s*t|
|
|
a\s*b\s*o\s*u\s*t)\s*:/ix) {
|
|
delete $hash->{$attr};
|
|
}
|
|
|
|
if ($attr eq 'style' && $opts->{'cleancss'}) {
|
|
# css2 spec, section 4.1.3
|
|
# position === p\osition :(
|
|
# strip all slashes no matter what.
|
|
$hash->{$attr} =~ s/\\//g;
|
|
# and catch the obvious ones.
|
|
foreach my $css (qw(absolute fixed)) {
|
|
if ($hash->{$attr} =~ /$css/i) {
|
|
delete $hash->{$attr};
|
|
last;
|
|
}
|
|
}
|
|
}
|
|
|
|
# reserve ljs_* ids for divs, etc so users can't override them to replace content
|
|
if ($attr eq 'id' && $hash->{$attr} =~ /^ljs_/i) {
|
|
delete $hash->{$attr};
|
|
}
|
|
|
|
if ($s1var) {
|
|
if ($attr =~ /%%/) {
|
|
delete $hash->{$attr};
|
|
next;
|
|
}
|
|
|
|
my $props = $LJ::S1::PROPS->{$s1var};
|
|
|
|
if ($hash->{$attr} =~ /^%%([\w:]+:)?(\S+?)%%$/ && $props->{$2} =~ /[aud]/) {
|
|
# don't change it.
|
|
} elsif ($hash->{$attr} =~ /^%%cons:\w+%%[^\%]*$/) {
|
|
# a site constant with something appended is also fine.
|
|
} elsif ($hash->{$attr} =~ /%%/) {
|
|
my $clean_var = sub {
|
|
my ($mods, $prop) = @_;
|
|
# HTML escape and kill line breaks
|
|
$mods = "attr:$mods" unless
|
|
$mods =~ /^(color|cons|siteroot|sitename|img):/ ||
|
|
$props->{$prop} =~ /[ud]/;
|
|
return '%%' . $mods . $prop . '%%';
|
|
};
|
|
|
|
$hash->{$attr} =~ s/[\n\r]//g;
|
|
$hash->{$attr} =~ s/%%([\w:]+:)?(\S+?)%%/$clean_var->(lc($1), $2)/eg;
|
|
|
|
if ($attr =~ /^(href|src|lowsrc|style)$/) {
|
|
$hash->{$attr} = "\%\%[attr[$hash->{$attr}]]\%\%";
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
if (exists $hash->{href}) {
|
|
unless ($hash->{href} =~ s/^lj:(?:\/\/)?(.*)$/ExpandLJURL($1)/ei) {
|
|
$hash->{href} = canonical_url($hash->{href}, 1);
|
|
}
|
|
}
|
|
if ($hash->{'style'} =~ /expression/i) {
|
|
delete $hash->{'style'};
|
|
}
|
|
|
|
if ($tag eq "img")
|
|
{
|
|
my $img_bad = 0;
|
|
if (defined $opts->{'maximgwidth'} &&
|
|
(! defined $hash->{'width'} ||
|
|
$hash->{'width'} > $opts->{'maximgwidth'})) { $img_bad = 1; }
|
|
if (defined $opts->{'maximgheight'} &&
|
|
(! defined $hash->{'height'} ||
|
|
$hash->{'height'} > $opts->{'maximgheight'})) { $img_bad = 1; }
|
|
if ($opts->{'extractimages'}) { $img_bad = 1; }
|
|
|
|
$hash->{src} = canonical_url($hash->{src}, 1);
|
|
|
|
if ($img_bad) {
|
|
$newdata .= "<a class=\"ljimgplaceholder\" href=\"" .
|
|
LJ::ehtml($hash->{'src'}) . "\">" .
|
|
LJ::img('placeholder') . '</a>';
|
|
$alt_output = 1;
|
|
}
|
|
}
|
|
|
|
if ($tag eq "a" && $extractlinks)
|
|
{
|
|
push @canonical_urls, canonical_url($token->[2]->{href}, 1);
|
|
$newdata .= "<b>";
|
|
next;
|
|
}
|
|
|
|
unless ($alt_output)
|
|
{
|
|
my $allow;
|
|
if ($mode eq "allow") {
|
|
$allow = 1;
|
|
if ($action{$tag} eq "deny") { $allow = 0; }
|
|
} else {
|
|
$allow = 0;
|
|
if ($action{$tag} eq "allow") { $allow = 1; }
|
|
}
|
|
|
|
if ($allow && ! $remove{$tag})
|
|
{
|
|
if ($opts->{'tablecheck'}) {
|
|
|
|
$allow = 0 if
|
|
|
|
# can't open table elements from outside a table
|
|
($tag =~ /^(?:tbody|thead|tfoot|tr|td|th)$/ && ! @tablescope) ||
|
|
|
|
# can't open td or th if not inside tr
|
|
($tag =~ /^(?:td|th)$/ && ! $tablescope[-1]->{'tr'}) ||
|
|
|
|
# can't open a table unless inside a td or th
|
|
($tag eq 'table' && @tablescope && ! grep { $tablescope[-1]->{$_} } qw(td th));
|
|
}
|
|
|
|
if ($allow) { $newdata .= "<$tag"; }
|
|
else { $newdata .= "<$tag"; }
|
|
|
|
# output attributes in original order, but only those
|
|
# that are allowed (by still being in %$hash after cleaning)
|
|
foreach (@$attrs) {
|
|
if ($hash->{$_} =~ /[^\x01-\x7f]/) {
|
|
# FIXME: this is so ghetto. make faster. make generic.
|
|
# HTML::Parser decodes entities for us (which is good)
|
|
# but in Perl 5.8 also includes the "poison" SvUTF8
|
|
# flag on the scalar it returns, thus poisoning the
|
|
# rest of the content this scalar is appended with.
|
|
# we need to remove that poison at this point. *sigh*
|
|
$hash->{$_} = pack('C*', unpack('C*', $hash->{$_}));
|
|
}
|
|
$newdata .= " $_=\"" . LJ::ehtml($hash->{$_}) . "\""
|
|
if exists $hash->{$_};
|
|
}
|
|
|
|
if ($slashclose) {
|
|
$newdata .= " /";
|
|
$opencount{$tag}--;
|
|
$tablescope[-1]->{$tag}-- if $opts->{'tablecheck'} && @tablescope;
|
|
}
|
|
if ($allow) {
|
|
$newdata .= ">";
|
|
$opencount{$tag}++;
|
|
|
|
# maintain current table scope
|
|
if ($opts->{'tablecheck'}) {
|
|
|
|
# open table
|
|
if ($tag eq 'table') {
|
|
push @tablescope, {};
|
|
|
|
# new tag within current table
|
|
} elsif (@tablescope) {
|
|
$tablescope[-1]->{$tag}++;
|
|
}
|
|
}
|
|
|
|
}
|
|
else { $newdata .= ">"; }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
# end tag
|
|
elsif ($type eq "E")
|
|
{
|
|
my $tag = $token->[1];
|
|
|
|
my $allow;
|
|
if ($tag eq "lj-raw") {
|
|
$opencount{$tag}--;
|
|
$tablescope[-1]->{$tag}-- if $opts->{'tablecheck'} && @tablescope;
|
|
}
|
|
elsif ($tag eq "lj-cut") {
|
|
if ($opts->{'cutpreview'}) {
|
|
$newdata .= "<b></lj-cut></b>";
|
|
}
|
|
} else {
|
|
if ($mode eq "allow") {
|
|
$allow = 1;
|
|
if ($action{$tag} eq "deny") { $allow = 0; }
|
|
} else {
|
|
$allow = 0;
|
|
if ($action{$tag} eq "allow") { $allow = 1; }
|
|
}
|
|
|
|
if ($extractlinks && $tag eq "a") {
|
|
if (@canonical_urls) {
|
|
my $url = LJ::ehtml(pop @canonical_urls);
|
|
$newdata .= "</b> ($url)";
|
|
}
|
|
}
|
|
|
|
if ($allow && ! $remove{$tag})
|
|
{
|
|
|
|
if ($opts->{'tablecheck'}) {
|
|
|
|
$allow = 0 if
|
|
|
|
# can't close table elements from outside a table
|
|
($tag =~ /^(?:table|tbody|thead|tfoot|tr|td|th)$/ && ! @tablescope) ||
|
|
|
|
# can't close td or th unless open tr
|
|
($tag =~ /^(?:td|th)$/ && ! $tablescope[-1]->{'tr'});
|
|
}
|
|
|
|
if ($allow && ! ($opts->{'noearlyclose'} && ! $opencount{$tag})) {
|
|
|
|
# maintain current table scope
|
|
if ($opts->{'tablecheck'}) {
|
|
|
|
# open table
|
|
if ($tag eq 'table') {
|
|
pop @tablescope;
|
|
|
|
# closing tag within current table
|
|
} elsif (@tablescope) {
|
|
$tablescope[-1]->{$tag}--;
|
|
}
|
|
}
|
|
|
|
$newdata .= "</$tag>";
|
|
$opencount{$tag}--;
|
|
|
|
} else { $newdata .= "</$tag>"; }
|
|
}
|
|
}
|
|
}
|
|
elsif ($type eq "D") {
|
|
# remove everything past first closing tag
|
|
$token->[1] =~ s/>.+/>/s;
|
|
# kill any opening tag except the starting one
|
|
$token->[1] =~ s/.<//sg;
|
|
$newdata .= $token->[1];
|
|
}
|
|
elsif ($type eq "T") {
|
|
my %url = ();
|
|
my $urlcount = 0;
|
|
|
|
if ($opencount{'style'}) {
|
|
# remove anything that might run javascript/vbscript code
|
|
# (Note: Ghetto. Need to use CSS.pm to build full tree, analyze, redump)
|
|
my $reduced = lc($token->[1]);
|
|
$reduced =~ s/\s+//g;
|
|
$reduced =~ s/\\//g;
|
|
$reduced =~ s/&\#(\d+);?/chr($1)/eg;
|
|
$reduced =~ s/&\#x(\w+);?/chr(hex($1))/eg;
|
|
$token->[1] = "/* potential scripting */"
|
|
if $reduced =~ /javascript|vbscript|expression/;
|
|
|
|
$token->[1] =~ s/<!--/[COMS]/g;
|
|
$token->[1] =~ s/-->/[COME]/g;
|
|
}
|
|
my $auto_format = $addbreaks &&
|
|
($opencount{'table'} <= ($opencount{'td'} + $opencount{'th'})) &&
|
|
! $opencount{'pre'} &&
|
|
! $opencount{'lj-raw'};
|
|
|
|
if ($auto_format && ! $noautolinks && ! $opencount{'a'} && ! $opencount{'textarea'}) {
|
|
my $match = sub {
|
|
my $str = shift;
|
|
if ($str =~ /^(.*?)(&(#39|quot|lt|gt)(;.*)?)$/) {
|
|
$url{++$urlcount} = $1;
|
|
return "&url$urlcount;$1&urlend;$2";
|
|
} else {
|
|
$url{++$urlcount} = $str;
|
|
return "&url$urlcount;$str&urlend;";
|
|
}
|
|
};
|
|
$token->[1] =~ s!https?://[^\s\'\"\<\>]+[a-zA-Z0-9_/&=\-]! $match->($&); !ge;
|
|
}
|
|
|
|
# escape tags in text tokens. shouldn't belong here!
|
|
# especially because the parser returns things it's
|
|
# confused about (broken, ill-formed HTML) as text.
|
|
$token->[1] =~ s/</</g;
|
|
unless ($opencount{'style'}) {
|
|
# don't escape this, because that breaks a CSS construct
|
|
$token->[1] =~ s/>/>/g;
|
|
}
|
|
if ($opencount{'style'}) {
|
|
$token->[1] =~ s/\[COMS\]/<!--/g;
|
|
$token->[1] =~ s/\[COME\]/-->/g;
|
|
}
|
|
|
|
# put <wbr> tags into long words, except inside <pre> and <textarea>.
|
|
if ($wordlength && !$opencount{'pre'} && !$opencount{'textarea'}) {
|
|
$token->[1] =~ s/\S{$wordlength,}/break_word($&,$wordlength)/eg;
|
|
}
|
|
|
|
# auto-format things, unless we're in a textarea, when it doesn't make sense
|
|
if ($auto_format && !$opencount{'textarea'}) {
|
|
$token->[1] =~ s/\r?\n/<br \/>/g;
|
|
if (! $opencount{'a'}) {
|
|
$token->[1] =~ s/&url(\d+);(.*?)&urlend;/<a href=\"$url{$1}\">$2<\/a>/g;
|
|
}
|
|
}
|
|
|
|
$newdata .= $token->[1];
|
|
}
|
|
elsif ($type eq "C") {
|
|
# by default, ditch comments
|
|
if ($keepcomments) {
|
|
my $com = $token->[1];
|
|
$com =~ s/^<!--\s*//;
|
|
$com =~ s/\s*--!>$//;
|
|
$com =~ s/<!--//;
|
|
$com =~ s/-->//;
|
|
$newdata .= "<!-- $com -->";
|
|
}
|
|
}
|
|
elsif ($type eq "PI") {
|
|
my $tok = $token->[1];
|
|
$tok =~ s/</</g;
|
|
$tok =~ s/>/>/g;
|
|
$newdata .= "<?$tok>";
|
|
}
|
|
else {
|
|
$newdata .= "<!-- OTHER: " . $type . "-->\n";
|
|
}
|
|
} # end while
|
|
|
|
# finish up open links if we're extracting them
|
|
if ($extractlinks && @canonical_urls) {
|
|
while (my $url = LJ::ehtml(pop @canonical_urls)) {
|
|
$newdata .= "</b> ($url)";
|
|
$opencount{'a'}--;
|
|
}
|
|
}
|
|
|
|
if (ref $opts->{'autoclose'} eq "ARRAY") {
|
|
foreach my $tag (@{$opts->{'autoclose'}}) {
|
|
if ($opencount{$tag}) {
|
|
$newdata .= "</$tag>" x $opencount{$tag};
|
|
}
|
|
}
|
|
}
|
|
|
|
# extra-paranoid check
|
|
1 while $newdata =~ s/<script\b//ig;
|
|
|
|
$$data = $newdata;
|
|
return undef;
|
|
}
|
|
|
|
|
|
# takes a reference to HTML and a base URL, and modifies HTML in place to use absolute URLs from the given base
|
|
sub resolve_relative_urls
|
|
{
|
|
my ($data, $base) = @_;
|
|
my $p = HTML::TokeParser->new($data);
|
|
|
|
# where we look for relative URLs
|
|
my $rel_source = {
|
|
'a' => {
|
|
'href' => 1,
|
|
},
|
|
'img' => {
|
|
'src' => 1,
|
|
},
|
|
};
|
|
|
|
my $global_did_mod = 0;
|
|
my $base_uri = undef; # until needed
|
|
my $newdata = "";
|
|
|
|
TOKEN:
|
|
while (my $token = $p->get_token)
|
|
{
|
|
my $type = $token->[0];
|
|
|
|
if ($type eq "S") # start tag
|
|
{
|
|
my $tag = $token->[1];
|
|
my $hash = $token->[2]; # attribute hashref
|
|
my $attrs = $token->[3]; # attribute names, in original order
|
|
|
|
my $did_mod = 0;
|
|
# see if this is a tag that could contain relative URLs we fix up.
|
|
if (my $relats = $rel_source->{$tag}) {
|
|
while (my $k = each %$relats) {
|
|
next unless defined $hash->{$k} && $hash->{$k} !~ /^[a-z]+:/;
|
|
my $rel_url = $hash->{$k};
|
|
$global_did_mod = $did_mod = 1;
|
|
|
|
$base_uri ||= URI->new($base);
|
|
$hash->{$k} = URI->new_abs($rel_url, $base_uri)->as_string;
|
|
}
|
|
}
|
|
|
|
# if no change was necessary
|
|
unless ($did_mod) {
|
|
$newdata .= $token->[4];
|
|
next TOKEN;
|
|
}
|
|
|
|
# otherwise, rebuild the opening tag
|
|
|
|
# for tags like <name/>, pretend it's <name> and reinsert the slash later
|
|
my $slashclose = 0; # If set to 1, use XML-style empty tag marker
|
|
$slashclose = 1 if $tag =~ s!/$!!;
|
|
$slashclose = 1 if delete $hash->{'/'};
|
|
|
|
# spit it back out
|
|
$newdata .= "<$tag";
|
|
# output attributes in original order
|
|
foreach (@$attrs) {
|
|
$newdata .= " $_=\"" . LJ::ehtml($hash->{$_}) . "\""
|
|
if exists $hash->{$_};
|
|
}
|
|
$newdata .= " /" if $slashclose;
|
|
$newdata .= ">";
|
|
}
|
|
elsif ($type eq "E") {
|
|
$newdata .= $token->[2];
|
|
}
|
|
elsif ($type eq "D") {
|
|
$newdata .= $token->[1];
|
|
}
|
|
elsif ($type eq "T") {
|
|
$newdata .= $token->[1];
|
|
}
|
|
elsif ($type eq "C") {
|
|
$newdata .= $token->[1];
|
|
}
|
|
elsif ($type eq "PI") {
|
|
$newdata .= $token->[2];
|
|
}
|
|
} # end while
|
|
|
|
$$data = $newdata if $global_did_mod;
|
|
return undef;
|
|
}
|
|
|
|
sub ExpandLJURL
|
|
{
|
|
my @args = grep { $_ } split(/\//, $_[0]);
|
|
my $mode = shift @args;
|
|
|
|
my %modes =
|
|
(
|
|
'faq' => sub {
|
|
my $id = shift()+0;
|
|
if ($id) {
|
|
return "support/faqbrowse.bml?faqid=$id";
|
|
} else {
|
|
return "support/faq.bml";
|
|
}
|
|
},
|
|
'memories' => sub {
|
|
my $user = LJ::canonical_username(shift);
|
|
if ($user) {
|
|
return "memories.bml?user=$user";
|
|
} else {
|
|
return "memories.bml";
|
|
}
|
|
},
|
|
'pubkey' => sub {
|
|
my $user = LJ::canonical_username(shift);
|
|
if ($user) {
|
|
return "pubkey.bml?user=$user";
|
|
} else {
|
|
return "pubkey.bml";
|
|
}
|
|
},
|
|
'support' => sub {
|
|
my $id = shift()+0;
|
|
if ($id) {
|
|
return "support/see_request.bml?id=$id";
|
|
} else {
|
|
return "support/";
|
|
}
|
|
},
|
|
'todo' => sub {
|
|
my $user = LJ::canonical_username(shift);
|
|
if ($user) {
|
|
return "todo/?user=$user";
|
|
} else {
|
|
return "todo/";
|
|
}
|
|
},
|
|
'user' => sub {
|
|
my $user = LJ::canonical_username(shift);
|
|
return "" if grep { /[\"\'\<\>\n\&]/ } @_;
|
|
return $_[0] eq 'profile' ?
|
|
"userinfo.bml?user=$user" :
|
|
"users/$user/" . join("", map { "$_/" } @_ );
|
|
},
|
|
'userinfo' => sub {
|
|
my $user = LJ::canonical_username(shift);
|
|
if ($user) {
|
|
return "userinfo.bml?user=$user";
|
|
} else {
|
|
return "userinfo.bml";
|
|
}
|
|
},
|
|
'userpics' => sub {
|
|
my $user = LJ::canonical_username(shift);
|
|
if ($user) {
|
|
return "allpics.bml?user=$user";
|
|
} else {
|
|
return "allpics.bml";
|
|
}
|
|
},
|
|
);
|
|
|
|
my $uri = $modes{$mode} ? $modes{$mode}->(@args) : "error:bogus-lj-url";
|
|
|
|
return "$LJ::SITEROOT/$uri";
|
|
}
|
|
|
|
my $subject_eat = [qw[head title style layer iframe applet object]];
|
|
my $subject_allow = [qw[a b i u em strong cite]];
|
|
my $subject_remove = [qw[bgsound embed object caption link font noscript]];
|
|
sub clean_subject
|
|
{
|
|
my $ref = shift;
|
|
return unless $$ref =~ /[\<\>]/;
|
|
clean($ref, {
|
|
'wordlength' => 40,
|
|
'addbreaks' => 0,
|
|
'eat' => $subject_eat,
|
|
'mode' => 'deny',
|
|
'allow' => $subject_allow,
|
|
'remove' => $subject_remove,
|
|
'autoclose' => $subject_allow,
|
|
'noearlyclose' => 1,
|
|
});
|
|
}
|
|
|
|
## returns a pure text subject (needed in links, email headers, etc...)
|
|
my $subjectall_eat = [qw[head title style layer iframe applet object]];
|
|
sub clean_subject_all
|
|
{
|
|
my $ref = shift;
|
|
return unless $$ref =~ /[\<\>]/;
|
|
clean($ref, {
|
|
'wordlength' => 40,
|
|
'addbreaks' => 0,
|
|
'eat' => $subjectall_eat,
|
|
'mode' => 'deny',
|
|
'textonly' => 1,
|
|
'autoclose' => $subject_allow,
|
|
'noearlyclose' => 1,
|
|
});
|
|
}
|
|
|
|
my $event_eat = [qw[head title style layer iframe applet object xml]];
|
|
my $event_remove = [qw[bgsound embed object link body meta noscript]];
|
|
|
|
my @comment_close = qw(
|
|
a sub sup xmp bdo q span
|
|
b i u tt s strike big small font
|
|
abbr acronym cite code dfn em kbd samp strong var del ins
|
|
h1 h2 h3 h4 h5 h6 div blockquote address pre center
|
|
ul ol li dl dt dd
|
|
table tr td th tbody tfoot thead colgroup caption
|
|
marquee area map form textarea blink
|
|
);
|
|
my @comment_all = (@comment_close, "img", "br", "hr", "p", "col");
|
|
|
|
my $userbio_eat = $event_eat;
|
|
my $userbio_remove = $event_remove;
|
|
my @userbio_close = @comment_close;
|
|
|
|
sub clean_event
|
|
{
|
|
my ($ref, $opts) = @_;
|
|
|
|
# old prototype was passing in the ref and preformatted flag.
|
|
# now the second argument is a hashref of options, so convert it to support the old way.
|
|
unless (ref $opts eq "HASH") {
|
|
$opts = { 'preformatted' => $opts };
|
|
}
|
|
|
|
# fast path: no markup or URLs to linkify
|
|
if ($$ref !~ /\<|\>|http/ && ! $opts->{preformatted}) {
|
|
$$ref =~ s/\S{40,}/break_word($&,40)/eg;
|
|
$$ref =~ s/\r?\n/<br \/>/g;
|
|
return;
|
|
}
|
|
|
|
# slow path: need to be run it through the cleaner
|
|
clean($ref, {
|
|
'linkify' => 1,
|
|
'wordlength' => 40,
|
|
'addbreaks' => $opts->{'preformatted'} ? 0 : 1,
|
|
'cuturl' => $opts->{'cuturl'},
|
|
'cutpreview' => $opts->{'cutpreview'},
|
|
'eat' => $event_eat,
|
|
'mode' => 'allow',
|
|
'remove' => $event_remove,
|
|
'autoclose' => \@comment_close,
|
|
'cleancss' => 1,
|
|
'maximgwidth' => $opts->{'maximgwidth'},
|
|
'maximgheight' => $opts->{'maximgheight'},
|
|
'ljcut_disable' => $opts->{'ljcut_disable'},
|
|
'noearlyclose' => 1,
|
|
});
|
|
}
|
|
|
|
sub get_okay_comment_tags
|
|
{
|
|
return @comment_all;
|
|
}
|
|
|
|
|
|
# ref: scalarref of text to clean, gets cleaned in-place
|
|
# opts: either a hashref of opts:
|
|
# - preformatted: if true, don't insert breaks and auto-linkify
|
|
# - anon_comment: don't linkify things, and prevent <a> tags
|
|
# or, opts can just be a boolean scalar, which implies the performatted tag
|
|
sub clean_comment
|
|
{
|
|
my ($ref, $opts) = @_;
|
|
|
|
unless (ref $opts) {
|
|
$opts = { 'preformatted' => $opts };
|
|
}
|
|
|
|
# fast path: no markup or URLs to linkify
|
|
if ($$ref !~ /\<|\>|http/ && ! $opts->{preformatted}) {
|
|
$$ref =~ s/\S{40,}/break_word($&,40)/eg;
|
|
$$ref =~ s/\r?\n/<br \/>/g;
|
|
return;
|
|
}
|
|
|
|
# slow path: need to be run it through the cleaner
|
|
clean($ref, {
|
|
'linkify' => 1,
|
|
'wordlength' => 40,
|
|
'addbreaks' => $opts->{preformatted} ? 0 : 1,
|
|
'eat' => [qw[head title style layer iframe applet object]],
|
|
'mode' => 'deny',
|
|
'allow' => \@comment_all,
|
|
'autoclose' => \@comment_close,
|
|
'cleancss' => 1,
|
|
'extractlinks' => $opts->{'anon_comment'},
|
|
'extractimages' => $opts->{'anon_comment'},
|
|
'noearlyclose' => 1,
|
|
});
|
|
}
|
|
|
|
sub clean_userbio {
|
|
my $ref = shift;
|
|
return undef unless ref $ref;
|
|
|
|
clean($ref, {
|
|
'wordlength' => 100,
|
|
'addbreaks' => 1,
|
|
'attrstrip' => [qw[style]],
|
|
'mode' => 'allow',
|
|
'noearlyclose' => 1,
|
|
'tablecheck' => 1,
|
|
'eat' => $userbio_eat,
|
|
'remove' => $userbio_remove,
|
|
'autoclose' => \@userbio_close,
|
|
});
|
|
}
|
|
|
|
sub clean_s1_style
|
|
{
|
|
my $s1 = shift;
|
|
my $clean;
|
|
|
|
my %tmpl;
|
|
LJ::parse_vars(\$s1, \%tmpl);
|
|
foreach my $v (keys %tmpl) {
|
|
clean(\$tmpl{$v}, {
|
|
'eat' => [qw[layer iframe script object embed applet]],
|
|
'mode' => 'allow',
|
|
'keepcomments' => 1, # allows CSS to work
|
|
's1var' => $v,
|
|
});
|
|
}
|
|
|
|
return Storable::freeze(\%tmpl);
|
|
}
|
|
|
|
sub s1_attribute_clean {
|
|
my $a = $_[0];
|
|
$a =~ s/[\t\n]//g;
|
|
$a =~ s/\"/"/g;
|
|
$a =~ s/\'/&\#39;/g;
|
|
$a =~ s/</</g;
|
|
$a =~ s/>/>/g;
|
|
|
|
# IE sucks:
|
|
if ($a =~ /((?:(?:v\s*b)|(?:j\s*a\s*v\s*a))\s*s\s*c\s*r\s*i\s*p\s*t|
|
|
a\s*b\s*o\s*u\s*t)\s*:/ix) { return ""; }
|
|
return $a;
|
|
}
|
|
|
|
sub canonical_url {
|
|
my $url = shift;
|
|
my $allow_all = shift;
|
|
|
|
# strip leading and trailing spaces
|
|
$url =~ s/^\s*//;
|
|
$url =~ s/\s*$//;
|
|
|
|
return unless $url;
|
|
|
|
unless ($allow_all) {
|
|
# see what protocol they want, default to http
|
|
my $pref = "http";
|
|
$pref = $1 if $url =~ /^(https?|ftp|webcal):/;
|
|
|
|
# strip out the protocol section
|
|
$url =~ s!^.*?:/*!!;
|
|
|
|
return unless $url;
|
|
|
|
# rebuild safe url
|
|
$url = "$pref://$url";
|
|
}
|
|
|
|
if ($LJ::FIXUP_AOL) {
|
|
# aol blocks http referred from lj, but ftp has no referer header.
|
|
if ($url =~ m!^http://(?:www\.)?(?:members|hometown|users)\.aol\.com/!) {
|
|
$url =~ s!^http!ftp!;
|
|
}
|
|
}
|
|
|
|
return $url;
|
|
}
|
|
|
|
sub break_word {
|
|
my ($word, $at) = @_;
|
|
$word =~ s/((?:$onechar){$at})\B/$1<wbr \/>/g;
|
|
return $word;
|
|
}
|
|
|
|
1;
|