3 * StatusNet, the distributed open-source microblogging tool
5 * Plugin to use bit.ly URL shortening services.
9 * LICENCE: This program is free software: you can redistribute it and/or modify
10 * it under the terms of the GNU Affero General Public License as published by
11 * the Free Software Foundation, either version 3 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU Affero General Public License for more details.
19 * You should have received a copy of the GNU Affero General Public License
20 * along with this program. If not, see <http://www.gnu.org/licenses/>.
24 * @author Craig Andrews <candrews@integralblue.com>
25 * @author Brion Vibber <brion@status.net>
26 * @copyright 2009 Free Software Foundation, Inc http://www.fsf.org
27 * @copyright 2010 StatusNet, Inc http://status.net/
28 * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
29 * @link http://status.net/
32 if (!defined('STATUSNET')) {
36 require_once INSTALLDIR.'/plugins/UrlShortener/UrlShortenerPlugin.php';
38 class BitlyUrlPlugin extends UrlShortenerPlugin
40 public $shortenerName = 'bit.ly';
41 public $serviceUrl = 'http://bit.ly/api?method=shorten&version=2.0.1&longUrl=%s';
42 public $login; // To set a site-default when admins or users don't override it.
45 function onInitializePlugin(){
46 parent::onInitializePlugin();
47 if(!isset($this->serviceUrl)){
48 throw new Exception(_m("You must specify a serviceUrl for bit.ly shortening."));
53 * Add bit.ly to the list of available URL shorteners if it's configured,
54 * otherwise leave it out.
56 * @param array $shorteners
57 * @return boolean hook return value
59 function onGetUrlShorteners(&$shorteners)
61 if ($this->getLogin() && $this->getApiKey()) {
62 return parent::onGetUrlShorteners($shorteners);
70 * @return string shortened version of the url, or null if URL shortening failed
72 protected function shorten($url) {
73 $response = $this->query($url);
74 if ($this->isOk($url, $response)) {
75 return $this->decode($url, $response->getBody());
82 * Get the user's or site-wide default bit.ly login name.
86 protected function getLogin()
88 $login = common_config('bitly', 'default_login');
90 $login = $this->login;
96 * Get the user's or site-wide default bit.ly API key.
100 protected function getApiKey()
102 $key = common_config('bitly', 'default_apikey');
104 $key = $this->apiKey;
110 * Inject API key into query before sending out...
113 * @return HTTPResponse
115 protected function query($url)
117 // http://code.google.com/p/bitly-api/wiki/ApiDocumentation#/shorten
118 $params = http_build_query(array(
119 'login' => $this->getLogin(),
120 'apiKey' => $this->getApiKey()), '', '&');
121 $serviceUrl = sprintf($this->serviceUrl, urlencode($url)) . '&' . $params;
123 $request = HTTPClient::start();
124 return $request->get($serviceUrl);
128 * JSON decode for API result
130 protected function decode($url, $body)
132 $json = json_decode($body, true);
133 return $json['results'][$url]['shortUrl'];
137 * JSON decode for API result
139 protected function isOk($url, $response)
143 if ($response->isOk()) {
144 $body = $response->getBody();
145 common_log(LOG_INFO, $body);
146 $json = json_decode($body, true);
147 if ($json['statusCode'] == 'OK') {
148 if (!isset($json['results'][$url])) {
149 common_log(LOG_ERR, "bit.ly returned OK response, but didn't find expected URL $url in $body");
152 $data = $json['results'][$url];
153 if (isset($data['shortUrl'])) {
155 } else if (isset($data['statusCode']) && $data['statusCode'] == 'ERROR') {
156 $code = $data['errorCode'];
157 $msg = $data['errorMessage'];
159 } else if ($json['statusCode'] == 'ERROR') {
160 $code = $json['errorCode'];
161 $msg = $json['errorMessage'];
163 common_log(LOG_ERR, "bit.ly returned error $code $msg for $url");
168 function onPluginVersion(&$versions)
170 $versions[] = array('name' => sprintf('BitlyUrl (%s)', $this->shortenerName),
171 'version' => STATUSNET_VERSION,
172 'author' => 'Craig Andrews, Brion Vibber',
173 'homepage' => 'http://status.net/wiki/Plugin:BitlyUrl',
175 sprintf(_m('Uses <a href="http://%1$s/">%1$s</a> URL-shortener service.'),
176 $this->shortenerName));
182 * Hook for RouterInitialized event.
184 * @param Net_URL_Mapper $m path-to-action mapper
185 * @return boolean hook return
187 function onRouterInitialized($m)
189 $m->connect('admin/bitly',
190 array('action' => 'bitlyadminpanel'));
195 * If the plugin's installed, this should be accessible to admins.
197 function onAdminPanelCheck($name, &$isOK)
199 if ($name == 'bitly') {
208 * Add the bit.ly admin panel to the list...
210 function onEndAdminPanelNav($nav)
212 if (AdminPanelAction::canAdmin('bitly')) {
213 $action_name = $nav->action->trimmed('action');
215 $nav->out->menuItem(common_local_url('bitlyadminpanel'),
217 _m('bit.ly URL shortening'),
218 $action_name == 'bitlyadminpanel',
219 'nav_bitly_admin_panel');
226 * Automatically load the actions and libraries used by the plugin
228 * @param Class $cls the class
230 * @return boolean hook return
233 function onAutoload($cls)
235 $base = dirname(__FILE__);
236 $lower = strtolower($cls);
238 case 'bitlyadminpanelaction':
239 require_once "$base/$lower.php";
247 * Internal hook point to check the default global credentials so
248 * the admin form knows if we have a fallback or not.
250 * @param string $login
251 * @param string $apiKey
252 * @return boolean hook return value
254 function onBitlyDefaultCredentials(&$login, &$apiKey)
256 $login = $this->login;
257 $apiKey = $this->apiKey;