Merge remote branch 'upstream/master'
[friendica-addons.git] / twitter / twitter.php
1 <?php
2 /**
3  * Name: Twitter Connector
4  * Description: Relay public postings to a connected Twitter account
5  * Version: 1.0.4
6  * Author: Tobias Diekershoff <http://diekershoff.homeunix.net/friendika/profile/tobias>
7  */
8
9
10 /*   Twitter Plugin for Friendica
11  *
12  *   Author: Tobias Diekershoff
13  *           tobias.diekershoff@gmx.net
14  *
15  *   License:3-clause BSD license
16  *
17  *   Configuration:
18  *     To use this plugin you need a OAuth Consumer key pair (key & secret)
19  *     you can get it from Twitter at https://twitter.com/apps
20  *
21  *     Register your Friendica site as "Client" application with "Read & Write" access
22  *     we do not need "Twitter as login". When you've registered the app you get the
23  *     OAuth Consumer key and secret pair for your application/site.
24  *
25  *     Add this key pair to your global .htconfig.php or use the admin panel.
26  *
27  *     $a->config['twitter']['consumerkey'] = 'your consumer_key here';
28  *     $a->config['twitter']['consumersecret'] = 'your consumer_secret here';
29  *
30  *     To activate the plugin itself add it to the $a->config['system']['addon']
31  *     setting. After this, your user can configure their Twitter account settings
32  *     from "Settings -> Plugin Settings".
33  *
34  *     Requirements: PHP5, curl [Slinky library]
35  *
36  *     Documentation: http://diekershoff.homeunix.net/redmine/wiki/friendikaplugin/Twitter_Plugin
37  */
38
39 function twitter_install() {
40         //  we need some hooks, for the configuration and for sending tweets
41         register_hook('connector_settings', 'addon/twitter/twitter.php', 'twitter_settings'); 
42         register_hook('connector_settings_post', 'addon/twitter/twitter.php', 'twitter_settings_post');
43         register_hook('post_local', 'addon/twitter/twitter.php', 'twitter_post_local');
44         register_hook('notifier_normal', 'addon/twitter/twitter.php', 'twitter_post_hook');
45         register_hook('jot_networks', 'addon/twitter/twitter.php', 'twitter_jot_nets');
46         logger("installed twitter");
47 }
48
49
50 function twitter_uninstall() {
51         unregister_hook('connector_settings', 'addon/twitter/twitter.php', 'twitter_settings'); 
52         unregister_hook('connector_settings_post', 'addon/twitter/twitter.php', 'twitter_settings_post');
53         unregister_hook('post_local', 'addon/twitter/twitter.php', 'twitter_post_local');
54         unregister_hook('notifier_normal', 'addon/twitter/twitter.php', 'twitter_post_hook');
55         unregister_hook('jot_networks', 'addon/twitter/twitter.php', 'twitter_jot_nets');
56
57         // old setting - remove only
58         unregister_hook('post_local_end', 'addon/twitter/twitter.php', 'twitter_post_hook');
59         unregister_hook('plugin_settings', 'addon/twitter/twitter.php', 'twitter_settings'); 
60         unregister_hook('plugin_settings_post', 'addon/twitter/twitter.php', 'twitter_settings_post');
61
62 }
63
64 function twitter_jot_nets(&$a,&$b) {
65         if(! local_user())
66                 return;
67
68         $tw_post = get_pconfig(local_user(),'twitter','post');
69         if(intval($tw_post) == 1) {
70                 $tw_defpost = get_pconfig(local_user(),'twitter','post_by_default');
71                 $selected = ((intval($tw_defpost) == 1) ? ' checked="checked" ' : '');
72                 $b .= '<div class="profile-jot-net"><input type="checkbox" name="twitter_enable"' . $selected . ' value="1" /> ' 
73                         . t('Post to Twitter') . '</div>';      
74         }
75
76
77 }
78
79 function twitter_settings_post ($a,$post) {
80         if(! local_user())
81                 return;
82         // don't check twitter settings if twitter submit button is not clicked 
83         if (!x($_POST,'twitter-submit')) return;
84         
85         if (isset($_POST['twitter-disconnect'])) {
86                 /***
87                  * if the twitter-disconnect checkbox is set, clear the OAuth key/secret pair
88                  * from the user configuration
89                  */
90                 del_pconfig( local_user(), 'twitter', 'consumerkey'  );
91                 del_pconfig( local_user(), 'twitter', 'consumersecret' );
92                 del_pconfig( local_user(), 'twitter', 'oauthtoken'  );  
93                 del_pconfig( local_user(), 'twitter', 'oauthsecret'  );  
94                 del_pconfig( local_user(), 'twitter', 'post' );
95                 del_pconfig( local_user(), 'twitter', 'post_by_default' );
96                 del_pconfig( local_user(), 'twitter', 'post_taglinks');
97         } else {
98         if (isset($_POST['twitter-pin'])) {
99                 //  if the user supplied us with a PIN from Twitter, let the magic of OAuth happen
100                 logger('got a Twitter PIN');
101                 require_once('library/twitteroauth.php');
102                 $ckey    = get_config('twitter', 'consumerkey'  );
103                 $csecret = get_config('twitter', 'consumersecret' );
104                 //  the token and secret for which the PIN was generated were hidden in the settings
105                 //  form as token and token2, we need a new connection to Twitter using these token
106                 //  and secret to request a Access Token with the PIN
107                 $connection = new TwitterOAuth($ckey, $csecret, $_POST['twitter-token'], $_POST['twitter-token2']);
108                 $token   = $connection->getAccessToken( $_POST['twitter-pin'] );
109                 //  ok, now that we have the Access Token, save them in the user config
110                 set_pconfig(local_user(),'twitter', 'oauthtoken',  $token['oauth_token']);
111                 set_pconfig(local_user(),'twitter', 'oauthsecret', $token['oauth_token_secret']);
112                 set_pconfig(local_user(),'twitter', 'post', 1);
113                 set_pconfig(local_user(),'twitter', 'post_taglinks', 1);
114                 //  reload the Addon Settings page, if we don't do it see Bug #42
115                 goaway($a->get_baseurl().'/settings/connectors');
116         } else {
117                 //  if no PIN is supplied in the POST variables, the user has changed the setting
118                 //  to post a tweet for every new __public__ posting to the wall
119                 set_pconfig(local_user(),'twitter','post',intval($_POST['twitter-enable']));
120                 set_pconfig(local_user(),'twitter','post_by_default',intval($_POST['twitter-default']));
121                 set_pconfig(local_user(),'twitter','post_taglinks',intval($_POST['twitter-sendtaglinks']));
122                 info( t('Twitter settings updated.') . EOL);
123         }}
124 }
125 function twitter_settings(&$a,&$s) {
126         if(! local_user())
127                 return;
128         $a->page['htmlhead'] .= '<link rel="stylesheet"  type="text/css" href="' . $a->get_baseurl() . '/addon/twitter/twitter.css' . '" media="all" />' . "\r\n";
129         /***
130          * 1) Check that we have global consumer key & secret
131          * 2) If no OAuthtoken & stuff is present, generate button to get some
132          * 3) Checkbox for "Send public notices (140 chars only)
133          */
134         $ckey    = get_config('twitter', 'consumerkey' );
135         $csecret = get_config('twitter', 'consumersecret' );
136         $otoken  = get_pconfig(local_user(), 'twitter', 'oauthtoken'  );
137         $osecret = get_pconfig(local_user(), 'twitter', 'oauthsecret' );
138         $enabled = get_pconfig(local_user(), 'twitter', 'post');
139         $checked = (($enabled) ? ' checked="checked" ' : '');
140         $defenabled = get_pconfig(local_user(),'twitter','post_by_default');
141         $defchecked = (($defenabled) ? ' checked="checked" ' : '');
142         $linksenabled = get_pconfig(local_user(),'twitter','post_taglinks');
143         $linkschecked = (($linksenabled) ? ' checked="checked" ' : '');
144
145         $s .= '<div class="settings-block">';
146         $s .= '<h3>'. t('Twitter Posting Settings') .'</h3>';
147
148         if ( (!$ckey) && (!$csecret) ) {
149                 /***
150                  * no global consumer keys
151                  * display warning and skip personal config
152                  */
153                 $s .= '<p>'. t('No consumer key pair for Twitter found. Please contact your site administrator.') .'</p>';
154         } else {
155                 /***
156                  * ok we have a consumer key pair now look into the OAuth stuff
157                  */
158                 if ( (!$otoken) && (!$osecret) ) {
159                         /***
160                          * the user has not yet connected the account to twitter...
161                          * get a temporary OAuth key/secret pair and display a button with
162                          * which the user can request a PIN to connect the account to a
163                          * account at Twitter.
164                          */
165                         require_once('library/twitteroauth.php');
166                         $connection = new TwitterOAuth($ckey, $csecret);
167                         $request_token = $connection->getRequestToken();
168                         $token = $request_token['oauth_token'];
169                         /***
170                          *  make some nice form
171                          */
172                         $s .= '<p>'. t('At this Friendica instance the Twitter plugin was enabled but you have not yet connected your account to your Twitter account. To do so click the button below to get a PIN from Twitter which you have to copy into the input box below and submit the form. Only your <strong>public</strong> posts will be posted to Twitter.') .'</p>';
173                         $s .= '<a href="'.$connection->getAuthorizeURL($token).'" target="_twitter"><img src="addon/twitter/lighter.png" alt="'.t('Log in with Twitter').'"></a>';
174                         $s .= '<div id="twitter-pin-wrapper">';
175                         $s .= '<label id="twitter-pin-label" for="twitter-pin">'. t('Copy the PIN from Twitter here') .'</label>';
176                         $s .= '<input id="twitter-pin" type="text" name="twitter-pin" />';
177                         $s .= '<input id="twitter-token" type="hidden" name="twitter-token" value="'.$token.'" />';
178                         $s .= '<input id="twitter-token2" type="hidden" name="twitter-token2" value="'.$request_token['oauth_token_secret'].'" />';
179             $s .= '</div><div class="clear"></div>';
180             $s .= '<div class="settings-submit-wrapper" ><input type="submit" name="twitter-submit" class="settings-submit" value="' . t('Submit') . '" /></div>';
181                 } else {
182                         /***
183                          *  we have an OAuth key / secret pair for the user
184                          *  so let's give a chance to disable the postings to Twitter
185                          */
186                         require_once('library/twitteroauth.php');
187                         $connection = new TwitterOAuth($ckey,$csecret,$otoken,$osecret);
188                         $details = $connection->get('account/verify_credentials');
189                         $s .= '<div id="twitter-info" ><img id="twitter-avatar" src="'.$details->profile_image_url.'" /><p id="twitter-info-block">'. t('Currently connected to: ') .'<a href="https://twitter.com/'.$details->screen_name.'" target="_twitter">'.$details->screen_name.'</a><br /><em>'.$details->description.'</em></p></div>';
190                         $s .= '<p>'. t('If enabled all your <strong>public</strong> postings can be posted to the associated Twitter account. You can choose to do so by default (here) or for every posting separately in the posting options when writing the entry.') .'</p>';
191                         if ($a->user['hidewall']) {
192                             $s .= '<p>'. t('<strong>Note</strong>: Due your privacy settings (<em>Hide your profile details from unknown viewers?</em>) the link potentially included in public postings relayed to Twitter will lead the visitor to a blank page informing the visitor that the access to your profile has been restricted.') .'</p>';
193                         }
194                         $s .= '<div id="twitter-enable-wrapper">';
195                         $s .= '<label id="twitter-enable-label" for="twitter-checkbox">'. t('Allow posting to Twitter'). '</label>';
196                         $s .= '<input id="twitter-checkbox" type="checkbox" name="twitter-enable" value="1" ' . $checked . '/>';
197                         $s .= '<div class="clear"></div>';
198                         $s .= '<label id="twitter-default-label" for="twitter-default">'. t('Send public postings to Twitter by default') .'</label>';
199                         $s .= '<input id="twitter-default" type="checkbox" name="twitter-default" value="1" ' . $defchecked . '/>';
200                         $s .= '<div class="clear"></div>';
201                         $s .= '<label id="twitter-sendtaglinks-label" for="twitter-sendtaglinks">'.t('Send linked #-tags and @-names to Twitter').'</label>';
202                         $s .= '<input id="twitter-sendtaglinks" type="checkbox" name="twitter-sendtaglinks" value="1" '. $linkschecked . '/>';
203                         $s .= '</div><div class="clear"></div>';
204
205                         $s .= '<div id="twitter-disconnect-wrapper">';
206                         $s .= '<label id="twitter-disconnect-label" for="twitter-disconnect">'. t('Clear OAuth configuration') .'</label>';
207                         $s .= '<input id="twitter-disconnect" type="checkbox" name="twitter-disconnect" value="1" />';
208                         $s .= '</div><div class="clear"></div>';
209                         $s .= '<div class="settings-submit-wrapper" ><input type="submit" name="twitter-submit" class="settings-submit" value="' . t('Submit') . '" /></div>'; 
210                 }
211         }
212         $s .= '</div><div class="clear"></div>';
213 }
214
215
216 function twitter_post_local(&$a,&$b) {
217
218         if($b['edit'])
219                 return;
220
221         if((local_user()) && (local_user() == $b['uid']) && (! $b['private']) && (! $b['parent']) ) {
222
223                 $twitter_post = intval(get_pconfig(local_user(),'twitter','post'));
224                 $twitter_enable = (($twitter_post && x($_REQUEST,'twitter_enable')) ? intval($_REQUEST['twitter_enable']) : 0);
225
226                 // if API is used, default to the chosen settings
227                 if($_REQUEST['api_source'] && intval(get_pconfig(local_user(),'twitter','post_by_default')))
228                         $twitter_enable = 1;
229
230         if(! $twitter_enable)
231             return;
232
233         if(strlen($b['postopts']))
234             $b['postopts'] .= ',';
235         $b['postopts'] .= 'twitter';
236         }
237 }
238
239 if (! function_exists('short_link')) {
240 function short_link ($url) {
241     require_once('library/slinky.php');
242     $slinky = new Slinky( $url );
243     $yourls_url = get_config('yourls','url1');
244     if ($yourls_url) {
245             $yourls_username = get_config('yourls','username1');
246             $yourls_password = get_config('yourls', 'password1');
247             $yourls_ssl = get_config('yourls', 'ssl1');
248             $yourls = new Slinky_YourLS();
249             $yourls->set( 'username', $yourls_username );
250             $yourls->set( 'password', $yourls_password );
251             $yourls->set( 'ssl', $yourls_ssl );
252             $yourls->set( 'yourls-url', $yourls_url );
253             $slinky->set_cascade( array( $yourls, new Slinky_UR1ca(), new Slinky_Trim(), new Slinky_IsGd(), new Slinky_TinyURL() ) );
254     }
255     else {
256             // setup a cascade of shortening services
257             // try to get a short link from these services
258             // in the order ur1.ca, trim, id.gd, tinyurl
259             $slinky->set_cascade( array( new Slinky_UR1ca(), new Slinky_Trim(), new Slinky_IsGd(), new Slinky_TinyURL() ) );
260     }
261     return $slinky->short();
262 } };
263
264 function twitter_post_hook(&$a,&$b) {
265
266         /**
267          * Post to Twitter
268          */
269
270         if($b['deleted'] || $b['private'] || ($b['created'] !== $b['edited']))
271         return;
272
273         if(! strstr($b['postopts'],'twitter'))
274                 return;
275
276         if($b['parent'] != $b['id'])
277                 return;
278
279         logger('twitter post invoked');
280
281
282         load_pconfig($b['uid'], 'twitter');
283
284         $ckey    = get_config('twitter', 'consumerkey'  );
285         $csecret = get_config('twitter', 'consumersecret' );
286         $otoken  = get_pconfig($b['uid'], 'twitter', 'oauthtoken'  );
287         $osecret = get_pconfig($b['uid'], 'twitter', 'oauthsecret' );
288
289         if($ckey && $csecret && $otoken && $osecret) {
290                 logger('twitter: we have customer key and oauth stuff, going to send.', LOGGER_DEBUG);
291
292                 require_once('library/twitteroauth.php');
293                 require_once('include/bbcode.php');
294                 $tweet = new TwitterOAuth($ckey,$csecret,$otoken,$osecret);
295                 // in theory max char is 140 but T. uses t.co to make links 
296                 // longer so we give them 10 characters extra
297                 $max_char = 130; // max. length for a tweet
298                 // we will only work with up to two times the length of the dent 
299                 // we can later send to Twitter. This way we can "gain" some 
300                 // information during shortening of potential links but do not 
301                 // shorten all the links in a 200000 character long essay.
302                 if (! $b['title']=='') {
303                     $tmp = $b['title'] . ' : '. $b['body'];
304 //                    $tmp = substr($tmp, 0, 4*$max_char);
305                 } else {
306                     $tmp = $b['body']; // substr($b['body'], 0, 3*$max_char);
307                 }
308                 // if [url=bla][img]blub.png[/img][/url] get blub.png
309                 $tmp = preg_replace( '/\[url\=(https?\:\/\/[a-zA-Z0-9\:\/\-\?\&\;\.\=\_\~\#\%\$\!\+\,]+)\]\[img\](\\w+.*?)\\[\\/img\]\\[\\/url\]/i', '$2', $tmp);
310                 // preserve links to images, videos and audios
311                 $tmp = preg_replace( '/\[img\=([0-9]*)x([0-9]*)\](.*?)\[\/img\]/ism', '$3', $tmp);
312                 $tmp = preg_replace( '/\[\\/?img(\\s+.*?\]|\])/i', '', $tmp);
313                 $tmp = preg_replace( '/\[\\/?video(\\s+.*?\]|\])/i', '', $tmp);
314                 $tmp = preg_replace( '/\[\\/?youtube(\\s+.*?\]|\])/i', '', $tmp);
315                 $tmp = preg_replace( '/\[\\/?vimeo(\\s+.*?\]|\])/i', '', $tmp);
316                 $tmp = preg_replace( '/\[\\/?audio(\\s+.*?\]|\])/i', '', $tmp);
317                 $linksenabled = get_pconfig($b['uid'],'twitter','post_taglinks');
318                 // if a #tag is linked, don't send the [url] over to SN
319                 // that is, don't send if the option is not set in the
320                 // connector settings
321                 if ($linksenabled=='0') {
322                         // #-tags
323                         $tmp = preg_replace( '/#\[url\=(\w+.*?)\](\w+.*?)\[\/url\]/i', '#$2', $tmp);
324                         // @-mentions
325                         $tmp = preg_replace( '/@\[url\=(\w+.*?)\](\w+.*?)\[\/url\]/i', '@$2', $tmp);
326                         // recycle 1
327                         $recycle = html_entity_decode("&#x2672; ", ENT_QUOTES, 'UTF-8');
328                         $tmp = preg_replace( '/'.$recycle.'\[url\=(\w+.*?)\](\w+.*?)\[\/url\]/i', $recycle.'$2', $tmp);
329                         // recycle 2 (Test)
330                         $recycle = html_entity_decode("&#x25CC; ", ENT_QUOTES, 'UTF-8');
331                         $tmp = preg_replace( '/'.$recycle.'\[url\=(\w+.*?)\](\w+.*?)\[\/url\]/i', $recycle.'$2', $tmp);
332                 }
333                 $tmp = preg_replace( '/\[url\=(https?\:\/\/[a-zA-Z0-9\:\/\-\?\&\;\.\=\_\~\#\%\$\!\+\,]+)\](\w+.*?)\[\/url\]/i', '$2 $1', $tmp);
334                 $tmp = preg_replace( '/\[bookmark\=(https?\:\/\/[a-zA-Z0-9\:\/\-\?\&\;\.\=\_\~\#\%\$\!\+\,]+)\](\w+.*?)\[\/bookmark\]/i', '$2 $1', $tmp);
335                 // find all http or https links in the body of the entry and
336                 // apply the shortener if the link is longer then 20 characters
337                 if (( strlen($tmp)>$max_char ) && ( $max_char > 0 )) {
338                     preg_match_all ( '/(https?\:\/\/[a-zA-Z0-9\:\/\-\?\&\;\.\=\_\~\#\%\$\!\+\,]+)/i', $tmp, $allurls  );
339                     foreach ($allurls as $url) {
340                         foreach ($url as $u) {
341                             if (strlen($u)>20) {
342                                 $sl = short_link($u);
343                                 $tmp = str_replace( $u, $sl, $tmp );
344                             }
345                         }
346                     }
347                 }
348                 // ok, all the links we want to send out are save, now strip 
349                 // away the remaining bbcode
350                 //$msg = strip_tags(bbcode($tmp, false, false));
351                 $msg = bbcode($tmp, false, false);
352                 $msg = str_replace(array('<br>','<br />'),"\n",$msg);
353                 $msg = strip_tags($msg);
354
355                 // quotes not working - let's try this
356                 $msg = html_entity_decode($msg);
357                 if (( strlen($msg) > $max_char) && $max_char > 0) {
358                         $shortlink = short_link( $b['plink'] );
359                         // the new message will be shortened such that "... $shortlink"
360                         // will fit into the character limit
361                         $msg = nl2br(substr($msg, 0, $max_char-strlen($shortlink)-4));
362                         $msg = str_replace(array('<br>','<br />'),' ',$msg);
363                         $e = explode(' ', $msg);
364                         //  remove the last word from the cut down message to 
365                         //  avoid sending cut words to the MicroBlog
366                         array_pop($e);
367                         $msg = implode(' ', $e);
368                         $msg .= '... ' . $shortlink;
369                 }
370
371                 $msg = trim($msg);
372
373                 // and now tweet it :-)
374                 if(strlen($msg)) {
375                         $result = $tweet->post('statuses/update', array('status' => $msg));
376                         logger('twitter_post send, result: ' . print_r($result, true), LOGGER_DEBUG);
377                         if ($result->error) {
378                                 logger('Send to Twitter failed: "' . $result->error . '"');
379                         }
380                 }
381         }
382 }
383
384 function twitter_plugin_admin_post(&$a){
385         $consumerkey    =       ((x($_POST,'consumerkey'))              ? notags(trim($_POST['consumerkey']))   : '');
386         $consumersecret =       ((x($_POST,'consumersecret'))   ? notags(trim($_POST['consumersecret'])): '');
387         set_config('twitter','consumerkey',$consumerkey);
388         set_config('twitter','consumersecret',$consumersecret);
389         info( t('Settings updated.'). EOL );
390 }
391 function twitter_plugin_admin(&$a, &$o){
392         $t = file_get_contents( dirname(__file__). "/admin.tpl" );
393         $o = replace_macros($t, array(
394                 '$submit' => t('Submit'),
395                                                                 // name, label, value, help, [extra values]
396                 '$consumerkey' => array('consumerkey', t('Consumer key'),  get_config('twitter', 'consumerkey' ), ''),
397                 '$consumersecret' => array('consumersecret', t('Consumer secret'),  get_config('twitter', 'consumersecret' ), '')
398         ));
399 }