230 lines
6.7 KiB
Perl
Executable File
230 lines
6.7 KiB
Perl
Executable File
#!/usr/bin/perl
|
|
|
|
package TwitRSS;
|
|
|
|
use strict;
|
|
use warnings;
|
|
use utf8;
|
|
use 5.10.0;
|
|
use Data::Dumper;
|
|
use Readonly;
|
|
use HTML::TreeBuilder::XPath;
|
|
use HTML::TreeBuilder::LibXML;
|
|
use HTML::Entities qw(:DEFAULT encode_entities_numeric);
|
|
use LWP::ConnCache;
|
|
use LWP::UserAgent;
|
|
use LWP::Protocol::Net::Curl;
|
|
use CGI::Fast;
|
|
use Encode;
|
|
use POSIX qw(strftime);
|
|
use Exporter qw(import);
|
|
|
|
our @EXPORT = qw(fetch_user_feed fetch_search_feed items_from_feed display_feed err $TWITTER_BASEURL);
|
|
|
|
binmode STDOUT, 'utf8';
|
|
binmode STDIN, 'utf8';
|
|
|
|
HTML::TreeBuilder::LibXML->replace_original();
|
|
|
|
Readonly our $TWITTER_BASEURL => 'https://twitter.com';
|
|
Readonly my $MAX_AGE => 3600;
|
|
|
|
my $browser = LWP::UserAgent->new;
|
|
$browser->conn_cache(LWP::ConnCache->new(5));
|
|
$browser->timeout(2);
|
|
|
|
# fetch user feed
|
|
sub fetch_user_feed {
|
|
my $user = shift;
|
|
my $replies = shift;
|
|
my $url = "$TWITTER_BASEURL/$user";
|
|
$url .= "/with_replies" if $replies;
|
|
|
|
my $response = $browser->get($url);
|
|
unless ($response->is_success) {
|
|
err('Can’t screenscrape Twitter',404);
|
|
return;
|
|
}
|
|
return $response->content;
|
|
}
|
|
|
|
# fetch search feed
|
|
sub fetch_search_feed {
|
|
my $term = shift;
|
|
$term =~ s{^#}{};
|
|
$term =~ s{^%23}{};
|
|
|
|
my $url = "$TWITTER_BASEURL/search?f=tweets&vertical=default&src=typd&q=$term";
|
|
|
|
# TODO: Support hashtags
|
|
# if ($term =~ m{^#}) {
|
|
# $url = "$TWITTER_BASEURL/hashtag/$term?src=tren";
|
|
# }
|
|
|
|
my $response = $browser->get($url);
|
|
unless ($response->is_success) {
|
|
err('Can’t screenscrape Twitter',404);
|
|
return;
|
|
}
|
|
return $response->content;
|
|
}
|
|
|
|
# parse search or user feed content, build up items array
|
|
# assumption: twitter keeps its tweet markup the same in searches and user feeds
|
|
# if not split this into 2 subs
|
|
sub items_from_feed {
|
|
my $content = shift;
|
|
my @items;
|
|
|
|
my $tree= HTML::TreeBuilder::XPath->new;
|
|
$tree->parse($content);
|
|
my $tweets = $tree->findnodes( '//li' . class_contains('js-stream-item')); # new version 2015-06-02
|
|
if ($tweets) {
|
|
for my $li (@$tweets) {
|
|
my $tweet = $li->findnodes('./div'
|
|
. class_contains("js-stream-tweet")
|
|
)->[0]
|
|
;
|
|
next unless $tweet;
|
|
my $header = $tweet->findnodes('./div/div'
|
|
. class_contains("stream-item-header")
|
|
. "/a"
|
|
. class_contains("js-action-profile"))->[0];
|
|
my $bd = $tweet->findnodes( './div/div/p'
|
|
. class_contains("js-tweet-text")
|
|
)->[0];
|
|
my $body = "<![CDATA[" . encode_entities($bd->as_HTML,'^\n\x20-\x25\x27-\x7e"') . "]]>";
|
|
$body=~s{&(\w+);}{&$1;}gi;
|
|
$body=~s{href="/}{href="https://twitter.com/}gi; # add back in twitter.com to unbreak links to hashtags, users, etc.
|
|
$body=~s{<a[^>]+href="https://t.co[^"]+"[^>]+title="([^"]+)"[^>]*>}{ <a href="$1">}gi; # experimental! stop links going via t.co; if an a has a title use it as the href.
|
|
$body=~s{<a[^>]+title="([^"]+)"[^>]+href="https://t.co[^"]+"[^>]*>}{ <a href="$1">}gi; # experimental! stop links going via t.co; if an a has a title use it as the href.
|
|
$body=~s{target="_blank"}{}gi;
|
|
$body=~s{</?span[^>]*>}{}gi;
|
|
$body=~s{</?s[^>]*>}{}gi;
|
|
$body=~s{data-[\w\-]+="[^"]+"}{}gi; # validator doesn't like data-aria markup that we get from twitter
|
|
my $avatar = $header->findvalue('./img' . class_contains("avatar") . "/\@src");
|
|
my $fst_img_a = $tweet->findnodes( './div//div'
|
|
. class_contains("js-adaptive-photo"))->[0];
|
|
$fst_img_a = $tweet->findnodes( './div/div/div'
|
|
. class_contains("OldMedia")
|
|
. "/div/div/div")->[0] unless $fst_img_a;
|
|
my $fst_img="";
|
|
if($fst_img_a) {
|
|
$fst_img = $fst_img_a->findvalue('@data-image-url');
|
|
if($fst_img) {
|
|
$body=~s{\]\]>$}{"<img src=\"$fst_img\" width=\"250\" />\]\]>"}e;
|
|
}
|
|
}
|
|
my $fullname = $header->findvalue('./strong' . class_contains("fullname"));
|
|
my $username = $header->findvalue('./span' . class_contains("username"));
|
|
$username =~ s{<[^>]+>}{}g;
|
|
$username =~ s{^\s+}{};
|
|
$username =~ s{\s+$}{};
|
|
my $title = enctxt($bd->as_text);
|
|
$title=~s{ }{}gi;
|
|
$title=~s{http}{ http}; # links in title lose space
|
|
my $uri = $TWITTER_BASEURL . $tweet->findvalue('@data-permalink-path');
|
|
my $timestamp = $tweet->findnodes('./div/div'
|
|
. class_contains("stream-item-header")
|
|
. '/small/a'
|
|
. class_contains("tweet-timestamp"))->[0]->findvalue('./span/@data-time'
|
|
);
|
|
|
|
my $pub_date = strftime("%a, %d %b %Y %H:%M:%S %z", localtime($timestamp));
|
|
|
|
push @items, {
|
|
username => enctxt($username),
|
|
fullname => enctxt($fullname),
|
|
link => $uri,
|
|
guid => $uri,
|
|
title => $title,
|
|
description => $body,
|
|
timestamp => $timestamp,
|
|
pubDate => $pub_date,
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
$tree->delete;
|
|
err("Can't gather tweets for that search",404);
|
|
next;
|
|
}
|
|
$tree->delete;
|
|
@items;
|
|
}
|
|
|
|
# print an rss feed, with header
|
|
sub display_feed {
|
|
my $feed_url = shift;
|
|
my $feed_title = shift;
|
|
my $twitter_url = shift;
|
|
my @items = @_;
|
|
print<<ENDHEAD
|
|
Content-type: application/rss+xml
|
|
Cache-control: public, max-age=$MAX_AGE
|
|
Access-Control-Allow-Origin: *
|
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
<rss xmlns:atom="http://www.w3.org/2005/Atom" xmlns:georss="http://www.georss.org/georss" xmlns:twitter="http://api.twitter.com" xmlns:dc="http://purl.org/dc/elements/1.1/" version="2.0">
|
|
<channel>
|
|
<atom:link href="$feed_url" rel="self" type="application/rss+xml" />
|
|
<title>$feed_title</title>
|
|
<link>$twitter_url</link>
|
|
<description>$feed_title</description>
|
|
<language>en-us</language>
|
|
<ttl>40</ttl>
|
|
ENDHEAD
|
|
;
|
|
|
|
for (@items) {
|
|
print<<ENDITEM
|
|
<item>
|
|
<title>$_->{title}</title>
|
|
<dc:creator>$_->{fullname} ($_->{username})</dc:creator>
|
|
<description>$_->{description}</description>
|
|
<pubDate>$_->{pubDate}</pubDate>
|
|
<guid>$_->{guid}</guid>
|
|
<link>$_->{link}</link>
|
|
<twitter:source/>
|
|
<twitter:place/>
|
|
</item>
|
|
ENDITEM
|
|
;
|
|
}
|
|
|
|
print<<ENDRSS
|
|
</channel>
|
|
</rss>
|
|
ENDRSS
|
|
;
|
|
|
|
}
|
|
|
|
# convenience: Encode entities
|
|
sub enctxt {
|
|
my $text=shift;
|
|
encode_entities_numeric(decode_entities($text));
|
|
}
|
|
|
|
# convenience: Make XPATH class detection nicer
|
|
sub class_contains {
|
|
my $classname = shift;
|
|
"[contains(concat(' ',normalize-space(\@class),' '),' $classname ')]";
|
|
}
|
|
|
|
# exit, printing an error
|
|
sub err {
|
|
my ($msg,$status) = (shift,shift);
|
|
print<<ENDHEAD
|
|
Content-type: text/html
|
|
Status: $status
|
|
Cache-control: max-age=86400
|
|
Refresh: 10; url=http://twitrss.me
|
|
|
|
<html><head></head><body><h2>ERR: $msg</h2><p>Redirecting you back to <a href="http://twitrss.me">TwitRSS.me</a> in a few seconds. You might have spelled the username wrong or something</p></body></html>
|
|
ENDHEAD
|
|
;
|
|
}
|
|
|
|
1;
|