source: lab.git/Dev/twitter/twitterbot.pl

Last change on this file was 7848b0b, checked in by Ken-ichi Mito <mitty@…>, 11 years ago

enable SSL option for Net::Twitter::Lite::WithAPIv1_1

  • SSL is required
  • Property mode set to 100755
File size: 8.9 KB
Line 
1#! /usr/bin/perl -w
2
3use strict;
4use warnings;
5use utf8;
6
7## IMPORTANT ##
8# When Net::Twitter::Lite encounters a Twitter API error or a network error,
9# it throws a Net::Twitter::Lite::Error object.
10# You can catch and process these exceptions by using eval blocks and testing $@
11## from http://search.cpan.org/perldoc?Net::Twitter::Lite#ERROR_HANDLING
12use Net::Twitter::Lite::WithAPIv1_1;
13use FindBin qw($Bin);
14use YAML::Tiny;
15use Date::Parse qw(str2time);
16use Mail::Sendmail;
17use Encode;
18
19my $_execmode = $ARGV[0] || 0;
20sub VERBOSE () { $_execmode eq 'verbose' };
21sub DEBUG   () { VERBOSE or $_execmode eq 'debug' };
22use Data::Dumper;
23
24DEBUG and warn "$0: debug mode";
25
26my $conf = loadconf("$Bin/config.yml");
27if (! defined $conf) {
28    die "$0: cannot parse config file.\n";
29}
30my $stat = loadconf("$Bin/status.yml");
31if (! defined $stat) {
32    $stat = {};
33}
34
35my $bot = login($conf);
36if (! $bot->authorized) {
37    die "$0: this client is not yet authorized.\n";
38}
39
40my $tweets = {};
41%$tweets = (
42    %$tweets,
43    %{ or_search($bot, $conf->{hashtag}, $stat->{search}) }
44);
45%$tweets = (
46    %$tweets,
47    %{ mentions_ids($bot, $stat->{mention}) }
48);
49
50foreach my $id (sort keys %$tweets) {
51    # $tweets->{$id}{type} eq 'search'  => found by search API
52    #                      eq 'mention' => found by mention API
53    my $skip;
54    if ($tweets->{$id}{type} eq 'retweet') {
55        DEBUG and warn "skipping $id that was already retweeted\n";
56        $skip = 'retweet';
57    }
58    if (defined $conf->{allow}) {
59        $skip = 'allow';
60        foreach my $screen_name ( @{ $conf->{allow}{screen_name} } ) {
61            if ($tweets->{$id}{screen_name} eq $screen_name) {
62                DEBUG and warn "$id was allowed by screen_name\n";
63                undef $skip;
64                last;
65            }
66        }
67        foreach my $user_id ( @{ $conf->{allow}{user_id} } ) {
68            if ($tweets->{$id}{user_id} eq $user_id) {
69                DEBUG and warn "$id was allowed by user_id\n";
70                undef $skip;
71                last;
72            }
73        }
74    }
75   
76    if (defined $conf->{mail}) {
77        my $send;
78        foreach my $pickup ( @{ $conf->{mail}{pickup} } ) {
79            if ($tweets->{$id}{type} eq $pickup) {
80                $send = 1;
81                last;
82            }
83        }
84        if ($conf->{mail}{ignore_allowed}) {
85            if (defined $conf->{allow} and ! defined $skip) {
86                # this tweet was allowed to retweet, so that be ignored on mail
87                undef $send;
88            }
89        }
90       
91        if ($send) {
92            if (! exists($conf->{mail}{body}) ) {
93                $conf->{mail}{body} =
94                    "[localtime(date)] http://twitter.com/<screen_name>/status/<status_id> text\n" .
95                    "----------------------------------------\n"
96                ;
97            }
98            $conf->{mail}{body} .=
99                "[" . localtime($tweets->{$id}{date}) . "] " .
100                "http://twitter.com/" .
101                $tweets->{$id}{screen_name} . "/status/" .
102                $tweets->{$id}{status_id} . " " .
103                $tweets->{$id}{text} . "\n"
104            ;
105        }
106    }
107   
108    if ($skip) {
109        $stat->{$tweets->{$id}{type}} = $id;
110        next;
111    }
112   
113    DEBUG or sleep($conf->{sleep});
114   
115    # do retweet found tweets
116    my $res;
117    eval {
118        DEBUG  or $res = $bot->retweet($id);
119        DEBUG and warn "retweet($id) => ", Dumper($tweets->{$id});
120    };
121    if ($@) {
122        evalrescue($@);
123        warn "status_id => $id\n";
124        next;
125    }
126   
127    $stat->{$tweets->{$id}{type}} = $id;
128}
129
130if ($conf->{mail}{body}) {
131    my $body = encode("iso-2022-jp", $conf->{mail}{body});
132   
133    my %mail = (
134        Smtp            => $conf->{mail}{server},
135        From            => $conf->{mail}{from},
136        To              => join(", ", @{ $conf->{mail}{to} }),
137        Subject         => $conf->{mail}{subject},
138        "Content-Type"  => $conf->{mail}{contenttype},
139        Body            => $body,
140    );
141    DEBUG and warn "sending mail => ", Dumper(\%mail);
142   
143    DEBUG or sendmail(%mail) or warn "Error sending mail: $Mail::Sendmail::error\n";
144}
145
146if (ref $tweets and keys %{$tweets}) {
147    # save last status to yaml file
148    DEBUG  or YAML::Tiny::DumpFile("$Bin/status.yml", $stat);
149    DEBUG and warn "status.yml => ", Dumper($stat);
150}
151
152
153sub loadconf {
154    # load configration data from yaml formatted file
155    #   param   => scalar string of filename
156    #   ret     => hash object of yaml data
157   
158    my $file = shift @_;
159   
160    my $yaml = YAML::Tiny->read($file);
161   
162    if ($!) {
163        warn "$0: '$file' $!\n";
164    }
165   
166    DEBUG and warn "'$file' => ", Dumper($yaml);
167   
168    return $yaml->[0];
169}
170
171sub login {
172    # make Net::Twitter::Lite::WithAPIv1_1 object and login
173    #   param   => hash object of configration
174    #   ret     => Net::Twitter::Lite::WithAPIv1_1 object
175   
176    my $conf = shift @_;
177   
178    my $bot = Net::Twitter::Lite::WithAPIv1_1->new(
179        consumer_key    => $conf->{consumer_key},
180        consumer_secret => $conf->{consumer_secret},
181        legacy_lists_api => 0,
182        ssl             => 1,
183    );
184   
185    $bot->access_token($conf->{access_token});
186    $bot->access_token_secret($conf->{access_token_secret});
187   
188    return $bot;
189}
190
191sub or_search {
192    # search tweets containing keywords
193    #   param   => Net::Twitter::Lite::WithAPIv1_1 object, ArrayRef of keywords, since_id
194    #   ret     => HashRef of status_id (timeline order is destroyed)
195    #               or undef (none is found)
196   
197    my $bot      = shift @_;
198    my $keywords = shift @_;
199    my $since_id = shift @_ || 1;
200   
201    my $key = "";
202    foreach my $word (@$keywords) {
203        if ($key) {
204            $key .= " OR $word";
205        }
206        else {
207            $key = $word;
208        }
209    }
210    DEBUG and warn "searching '$key'";
211   
212    my $res;
213    my $ids = {};
214    eval {
215        if ($key) {
216            $res = $bot->search(
217                {
218                             => $key,
219                    since_id    => $since_id,
220                }
221            );
222        }
223        VERBOSE and warn Dumper($res);
224        if ($res->{statuses}) {
225            foreach my $tweet (@{$res->{statuses}}) {
226                my $res = $bot->show_status($tweet->{id});
227                VERBOSE and warn Dumper($res);
228               
229                my $id = {
230                    date        => str2time($res->{created_at}),
231                    screen_name => $res->{user}{screen_name},
232                    status_id   => $res->{id},
233                    text        => $res->{text},
234                    user_id     => $res->{user}{id},
235                };
236                if ($res->{retweeted_status}) {
237                    $id->{retweet_of}   = $res->{retweeted_status}{id};
238                    $id->{type}         = 'retweet';
239                }
240                else {
241                    $id->{type} = 'search';
242                }
243                $ids->{$tweet->{id}} = $id;
244            }
245        }
246    };
247    if ($@) {
248        evalrescue($@);
249    }
250   
251    DEBUG and warn "search result => ", Dumper($ids);
252    return $ids;
253}
254
255sub mentions_ids {
256    # return status_ids mentioned to me
257    #   param   => Net::Twitter::Lite::WithAPIv1_1 object, since_id
258    #   ret     => HashRef of status_id (timeline order is destroyed)
259    #               or undef (none is found)
260   
261    my $bot      = shift @_;
262    my $since_id = shift @_ || 1;
263   
264    my $res;
265    eval {
266        $res = $bot->mentions(
267            {
268                since_id    => $since_id,
269            }
270        );
271        VERBOSE and warn Dumper($res);
272    };
273    if ($@) {
274        evalrescue($@);
275    }
276   
277    my $ids = {};
278    if ($res && @{$res}) {
279        $ids = {
280            map {
281                $_->{id} => {
282                    date        => str2time($_->{created_at}),
283                    screen_name => $_->{user}{screen_name},
284                    status_id   => $_->{id},
285                    text        => $_->{text},
286                    type        => 'mention',
287                    user_id     => $_->{user}{id},
288                }
289            } @{$res}
290        };
291    }
292   
293    DEBUG and warn "mentions result => ", Dumper($ids);
294    return $ids;
295}
296
297sub evalrescue {
298    # output error message at eval error
299   
300    use Scalar::Util qw(blessed);
301   
302    if (blessed $@ && $@->isa('Net::Twitter::Lite::Error')) {
303        warn $@->error;
304        if ($@->twitter_error) {
305            my $twitter_error = $@->twitter_error;
306            if (defined $twitter_error->{errors} && ref($twitter_error->{errors})) {
307                foreach my $error (@{$twitter_error->{errors}}) {
308                    warn "code => "   , $error->{code}, "\n";
309                    warn "message => ", $error->{message}, "\n";
310                }
311            }
312            else {
313                # unknown HASH structure
314                use Data::Dumper;
315                warn Dumper $twitter_error;
316            }
317        }
318    }
319    else {
320        warn $@;
321    }
322}
Note: See TracBrowser for help on using the repository browser.