#!/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 = "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{]+href="https://t.co[^"]+"[^>]+title="([^"]+)"[^>]*>}{ }gi; # experimental! stop links going via t.co; if an a has a title use it as the href. $body=~s{]+title="([^"]+)"[^>]+href="https://t.co[^"]+"[^>]*>}{ }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{]*>}{}gi; $body=~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{\]\]>$}{"\]\]>"}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< $feed_title $twitter_url $feed_title en-us 40 ENDHEAD ; for (@items) { print< $_->{title} $_->{fullname} ($_->{username}) $_->{description} $_->{pubDate} $_->{guid} $_->{link} ENDITEM ; } print< 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<

ERR: $msg

Redirecting you back to TwitRSS.me in a few seconds. You might have spelled the username wrong or something

ENDHEAD ; } 1;