3 * Name: Buffer Post Connector
4 * Description: Post to Buffer (Facebook Pages, LinkedIn, Twitter)
6 * Author: Michael Vogel <http://pirati.ca/profile/heluecht>
9 require 'addon/buffer/bufferapp.php';
12 use Friendica\Core\Hook;
13 use Friendica\Core\Logger;
14 use Friendica\Core\Protocol;
15 use Friendica\Core\Renderer;
16 use Friendica\Database\DBA;
18 use Friendica\Model\ItemContent;
19 use Friendica\Util\Proxy as ProxyUtils;
20 use Friendica\Util\Strings;
22 function buffer_install()
24 Hook::register('hook_fork', 'addon/buffer/buffer.php', 'buffer_hook_fork');
25 Hook::register('post_local', 'addon/buffer/buffer.php', 'buffer_post_local');
26 Hook::register('notifier_normal', 'addon/buffer/buffer.php', 'buffer_send');
27 Hook::register('jot_networks', 'addon/buffer/buffer.php', 'buffer_jot_nets');
28 Hook::register('connector_settings', 'addon/buffer/buffer.php', 'buffer_settings');
29 Hook::register('connector_settings_post', 'addon/buffer/buffer.php', 'buffer_settings_post');
32 function buffer_module()
36 function buffer_content(App $a)
39 notice(DI::l10n()->t('Permission denied.') . EOL);
43 require_once "mod/settings.php";
46 if (isset($a->argv[1])) {
47 switch ($a->argv[1]) {
49 $o = buffer_connect($a);
53 $o = print_r($a->argv, true);
57 $o = buffer_connect($a);
63 function buffer_addon_admin(App $a, &$o)
65 $t = Renderer::getMarkupTemplate("admin.tpl", "addon/buffer/");
67 $o = Renderer::replaceMacros($t, [
68 '$submit' => DI::l10n()->t('Save Settings'),
69 // name, label, value, help, [extra values]
70 '$client_id' => ['client_id', DI::l10n()->t('Client ID'), DI::config()->get('buffer', 'client_id'), ''],
71 '$client_secret' => ['client_secret', DI::l10n()->t('Client Secret'), DI::config()->get('buffer', 'client_secret'), ''],
75 function buffer_addon_admin_post(App $a)
77 $client_id = (!empty($_POST['client_id']) ? Strings::escapeTags(trim($_POST['client_id'])) : '');
78 $client_secret = (!empty($_POST['client_secret']) ? Strings::escapeTags(trim($_POST['client_secret'])) : '');
80 DI::config()->set('buffer', 'client_id' , $client_id);
81 DI::config()->set('buffer', 'client_secret', $client_secret);
84 function buffer_connect(App $a)
86 if (isset($_REQUEST["error"])) {
87 $o = DI::l10n()->t('Error when registering buffer connection:')." ".$_REQUEST["error"];
91 // Start a session. This is necessary to hold on to a few keys the callback script will also need
94 // Define the needed keys
95 $client_id = DI::config()->get('buffer','client_id');
96 $client_secret = DI::config()->get('buffer','client_secret');
98 // The callback URL is the script that gets called after the user authenticates with buffer
99 $callback_url = DI::baseUrl()->get()."/buffer/connect";
101 $buffer = new BufferApp($client_id, $client_secret, $callback_url);
104 $o = '<a href="' . $buffer->get_login_url() . '">Connect to Buffer!</a>';
106 Logger::log("buffer_connect: authenticated");
107 $o = DI::l10n()->t("You are now authenticated to buffer. ");
108 $o .= '<br /><a href="' . DI::baseUrl()->get() . '/settings/connectors">' . DI::l10n()->t("return to the connector page") . '</a>';
109 DI::pConfig()->set(local_user(), 'buffer','access_token', $buffer->access_token);
115 function buffer_jot_nets(App $a, array &$jotnets_fields)
121 if (DI::pConfig()->get(local_user(), 'buffer', 'post')) {
122 $jotnets_fields[] = [
123 'type' => 'checkbox',
126 DI::l10n()->t('Post to Buffer'),
127 DI::pConfig()->get(local_user(), 'buffer', 'post_by_default')
133 function buffer_settings(App $a, &$s)
135 if (! local_user()) {
139 /* Add our stylesheet to the page so we can make our settings look nice */
141 DI::page()['htmlhead'] .= '<link rel="stylesheet" type="text/css" href="' . DI::baseUrl()->get() . '/addon/buffer/buffer.css' . '" media="all" />' . "\r\n";
143 /* Get the current state of our config variables */
145 $enabled = DI::pConfig()->get(local_user(),'buffer','post');
146 $checked = (($enabled) ? ' checked="checked" ' : '');
147 $css = (($enabled) ? '' : '-disabled');
149 $def_enabled = DI::pConfig()->get(local_user(),'buffer','post_by_default');
150 $def_checked = (($def_enabled) ? ' checked="checked" ' : '');
152 /* Add some HTML to the existing form */
154 $s .= '<span id="settings_buffer_inflated" class="settings-block fakelink" style="display: block;" onclick="openClose(\'settings_buffer_expanded\'); openClose(\'settings_buffer_inflated\');">';
155 $s .= '<img class="connector'.$css.'" src="images/buffer.png" /><h3 class="connector">'. DI::l10n()->t('Buffer Export').'</h3>';
157 $s .= '<div id="settings_buffer_expanded" class="settings-block" style="display: none;">';
158 $s .= '<span class="fakelink" onclick="openClose(\'settings_buffer_expanded\'); openClose(\'settings_buffer_inflated\');">';
159 $s .= '<img class="connector'.$css.'" src="images/buffer.png" /><h3 class="connector">'. DI::l10n()->t('Buffer Export').'</h3>';
162 $client_id = DI::config()->get("buffer", "client_id");
163 $client_secret = DI::config()->get("buffer", "client_secret");
164 $access_token = DI::pConfig()->get(local_user(), "buffer", "access_token");
166 $s .= '<div id="buffer-password-wrapper">';
168 if ($access_token == "") {
169 $s .= '<div id="buffer-authenticate-wrapper">';
170 $s .= '<a href="'.DI::baseUrl()->get().'/buffer/connect">'.DI::l10n()->t("Authenticate your Buffer connection").'</a>';
171 $s .= '</div><div class="clear"></div>';
173 $s .= '<div id="buffer-enable-wrapper">';
174 $s .= '<label id="buffer-enable-label" for="buffer-checkbox">' . DI::l10n()->t('Enable Buffer Post Addon') . '</label>';
175 $s .= '<input id="buffer-checkbox" type="checkbox" name="buffer" value="1" ' . $checked . '/>';
176 $s .= '</div><div class="clear"></div>';
178 $s .= '<div id="buffer-bydefault-wrapper">';
179 $s .= '<label id="buffer-bydefault-label" for="buffer-bydefault">' . DI::l10n()->t('Post to Buffer by default') . '</label>';
180 $s .= '<input id="buffer-bydefault" type="checkbox" name="buffer_bydefault" value="1" ' . $def_checked . '/>';
181 $s .= '</div><div class="clear"></div>';
183 $s .= '<div id="buffer-delete-wrapper">';
184 $s .= '<label id="buffer-delete-label" for="buffer-delete">' . DI::l10n()->t('Check to delete this preset') . '</label>';
185 $s .= '<input id="buffer-delete" type="checkbox" name="buffer_delete" value="1" />';
186 $s .= '</div><div class="clear"></div>';
188 // The callback URL is the script that gets called after the user authenticates with buffer
189 $callback_url = DI::baseUrl()->get() . '/buffer/connect';
191 $buffer = new BufferApp($client_id, $client_secret, $callback_url, $access_token);
193 $profiles = $buffer->go('/profiles');
194 if (is_array($profiles)) {
195 $s .= '<div id="buffer-accounts-wrapper">';
196 $s .= DI::l10n()->t("Posts are going to all accounts that are enabled by default:");
198 foreach ($profiles as $profile) {
199 if (!$profile->default)
202 //$s .= "<img src='".$profile->avatar_https."' width='16' />";
203 $s .= " ".$profile->formatted_username." (".$profile->formatted_service.")";
207 $s .= '</div><div class="clear"></div>';
212 $s .= '</div><div class="clear"></div>';
214 /* provide a submit button */
216 $s .= '<div class="settings-submit-wrapper" ><input type="submit" id="buffer-submit" name="buffer-submit" class="settings-submit" value="' . DI::l10n()->t('Save Settings') . '" /></div></div>';
220 function buffer_settings_post(App $a, array &$b)
222 if (!empty($_POST['buffer-submit'])) {
223 if (!empty($_POST['buffer_delete'])) {
224 DI::pConfig()->set(local_user(), 'buffer', 'access_token' , '');
225 DI::pConfig()->set(local_user(), 'buffer', 'post' , false);
226 DI::pConfig()->set(local_user(), 'buffer', 'post_by_default', false);
228 DI::pConfig()->set(local_user(), 'buffer', 'post' , intval($_POST['buffer'] ?? false));
229 DI::pConfig()->set(local_user(), 'buffer', 'post_by_default', intval($_POST['buffer_bydefault'] ?? false));
234 function buffer_post_local(App $a, array &$b)
236 if (!local_user() || (local_user() != $b['uid'])) {
240 $buffer_post = intval(DI::pConfig()->get(local_user(),'buffer','post'));
242 $buffer_enable = (($buffer_post && !empty($_REQUEST['buffer_enable'])) ? intval($_REQUEST['buffer_enable']) : 0);
244 if ($b['api_source'] && intval(DI::pConfig()->get(local_user(),'buffer','post_by_default'))) {
248 if (!$buffer_enable) {
252 if (strlen($b['postopts'])) {
253 $b['postopts'] .= ',';
256 $b['postopts'] .= 'buffer';
259 function buffer_hook_fork(&$a, &$b)
261 if ($b['name'] != 'notifier_normal') {
267 if ($post['deleted'] || $post['private'] || ($post['created'] !== $post['edited']) ||
268 !strstr($post['postopts'], 'buffer') || ($post['parent'] != $post['id'])) {
269 $b['execute'] = false;
274 function buffer_send(App $a, array &$b)
276 if ($b['deleted'] || $b['private'] || ($b['created'] !== $b['edited'])) {
280 if (!strstr($b['postopts'],'buffer')) {
284 if ($b['parent'] != $b['id']) {
288 // Dont't post if the post doesn't belong to us.
289 // This is a check for forum postings
290 $self = DBA::selectFirst('contact', ['id'], ['uid' => $b['uid'], 'self' => true]);
291 if ($b['contact-id'] != $self['id']) {
295 // if post comes from buffer don't send it back
296 //if($b['app'] == "Buffer")
299 $client_id = DI::config()->get("buffer", "client_id");
300 $client_secret = DI::config()->get("buffer", "client_secret");
301 $access_token = DI::pConfig()->get($b['uid'], "buffer","access_token");
305 $buffer = new BufferApp($client_id, $client_secret, $callback_url, $access_token);
307 $profiles = $buffer->go('/profiles');
308 if (is_array($profiles)) {
309 Logger::log("Will send these parameter ".print_r($b, true), Logger::DEBUG);
311 foreach ($profiles as $profile) {
312 if (!$profile->default)
317 switch ($profile->service) {
319 $send = ($b["extid"] != Protocol::FACEBOOK);
321 $includedlinks = false;
326 $send = ($b["extid"] != Protocol::TWITTER);
328 $includedlinks = true;
333 $send = ($b["extid"] != Protocol::LINKEDIN);
335 $includedlinks = true;
345 $post = ItemContent::getPlaintextPost($item, $limit, $includedlinks, $htmlmode);
346 Logger::log("buffer_send: converted message ".$b["id"]." result: ".print_r($post, true), Logger::DEBUG);
348 // The image proxy is used as a sanitizer. Buffer seems to be really picky about pictures
349 if (isset($post["image"])) {
350 $post["image"] = ProxyUtils::proxifyUrl($post["image"]);
353 if (isset($post["preview"])) {
354 $post["preview"] = ProxyUtils::proxifyUrl($post["preview"]);
357 // Seems like a bug to me
358 // Buffer doesn't add links to Twitter (but pictures)
359 if (($profile->service == "twitter") && isset($post["url"]) && ($post["type"] != "photo")) {
360 $post["text"] .= " " . $post["url"];
364 $message["text"] = $post["text"];
365 $message["profile_ids[]"] = $profile->id;
366 $message["shorten"] = false;
367 $message["now"] = true;
369 if (isset($post["title"])) {
370 $message["media[title]"] = $post["title"];
373 if (isset($post["description"])) {
374 $message["media[description]"] = $post["description"];
377 if (isset($post["url"]) && ($post["type"] != "photo")) {
378 $message["media[link]"] = $post["url"];
381 if (isset($post["image"])) {
382 $message["media[picture]"] = $post["image"];
384 if ($post["type"] == "photo") {
385 $message["media[thumbnail]"] = $post["image"];
389 if (isset($post["preview"])) {
390 $message["media[thumbnail]"] = $post["preview"];
394 Logger::log("buffer_send: data for message " . $b["id"] . ": " . print_r($message, true), Logger::DEBUG);
395 $ret = $buffer->go('/updates/create', $message);
396 Logger::log("buffer_send: send message " . $b["id"] . " result: " . print_r($ret, true), Logger::DEBUG);