* @license http://www.fsf.org/licensing/licenses/agpl.html AGPLv3 * @link http://status.net/ * * StatusNet - the distributed open-source microblogging tool * Copyright (C) 2009, StatusNet, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ if (!defined('STATUSNET')) { exit(1); } class RSSCloudRequestNotifyAction extends Action { /** * Initialization. * * @param array $args Web and URL arguments * * @return boolean false if user doesn't exist */ function prepare($args) { parent::prepare($args); $this->ip = $_SERVER['REMOTE_ADDR']; $this->port = $this->arg('port'); $this->path = $this->arg('path'); if ($this->path[0] != '/') { $this->path = '/' . $this->path; } $this->protocol = $this->arg('protocol'); $this->procedure = $this->arg('notifyProcedure'); $this->domain = $this->arg('domain'); $this->feeds = $this->getFeeds(); return true; } function handle($args) { parent::handle($args); if ($_SERVER['REQUEST_METHOD'] != 'POST') { $this->showResult(false, 'Request must be POST.'); return; } $missing = array(); if (empty($this->port)) { $missing[] = 'port'; } if (empty($this->path)) { $missing[] = 'path'; } if (empty($this->protocol)) { $missing[] = 'protocol'; } if (!isset($this->procedure)) { $missing[] = 'notifyProcedure'; } if (!empty($missing)) { $msg = 'The following parameters were missing from the request body: ' . implode(', ', $missing) . '.'; $this->showResult(false, $msg); return; } if (empty($this->feeds)) { $this->showResult(false, 'You must provide at least one valid profile feed url (url1, url2, url3 ... urlN).'); return; } // We have to validate everything before saving anything. // We only return one success or failure no matter how // many feeds the subscriber is trying to subscribe to foreach ($this->feeds as $feed) { if (!$this->validateFeed($feed)) { $msg = 'Feed subscription failed - Not a valid feed.'; $this->showResult(false, $msg); return; } if (!$this->testNotificationHandler($feed)) { $msg = 'Feed subscription failed - ' . 'notification handler doesn\'t respond correctly.'; $this->showResult(false, $msg); return; } } foreach ($this->feeds as $feed) { $this->saveSubscription($feed); } // XXX: What to do about deleting stale subscriptions? // 25 hours seems harsh. WordPress doesn't ever remove // subscriptions. $msg = 'Thanks for the subscription. ' . 'When the feed(s) update(s) we\'ll notify you.'; $this->showResult(true, $msg); } function validateFeed($feed) { $user = $this->userFromFeed($feed); if (empty($user)) { return false; } return true; } function getFeeds() { $feeds = array(); while (list($key, $feed) = each ($this->args)) { if (preg_match('/^url\d*$/', $key)) { $feeds[] = $feed; } } return $feeds; } function testNotificationHandler($feed) { common_debug("RSSCloudPlugin - testNotificationHandler()"); $notifyUrl = $this->getNotifyUrl(); $notifier = new RSSCloudNotifier(); if (isset($this->domain)) { // 'domain' param set, so we have to use GET and send a challenge common_log(LOG_INFO, 'Testing notification handler with challenge: ' . $notifyUrl); return $notifier->challenge($notifyUrl, $feed); } else { common_log(LOG_INFO, 'Testing notification handler: ' . $notifyUrl); return $notifier->postUpdate($notifyUrl, $feed); } } function getNotifyUrl() { if (isset($this->domain)) { return 'http://' . $this->domain . ':' . $this->port . $this->path; } else { return 'http://' . $this->ip . ':' . $this->port . $this->path; } } function userFromFeed($feed) { // We only do profile feeds // XXX: Add cloud element to RSS 1.0 feeds? $path = common_path('api/statuses/user_timeline/'); $valid = '%^' . $path . '(?.*)\.rss$%'; if (preg_match($valid, $feed, $matches)) { $user = User::staticGet('nickname', $matches['nickname']); if (!empty($user)) { return $user; } } return false; } function saveSubscription($feed) { $user = $this->userFromFeed($feed); $notifyUrl = $this->getNotifyUrl(); $sub = RSSCloudSubscription::getSubscription($user->id, $notifyUrl); if ($sub) { common_debug("Already subscribed to that!"); } else { $sub = new RSSCloudSubscription(); $sub->subscribed = $user->id; $sub->url = $notifyUrl; $sub->created = common_sql_now(); if (!$sub->insert()) { common_log_db_error($sub, 'INSERT', __FILE__); return false; } } return true; } function showResult($success, $msg) { $this->startXML(); $this->elementStart('notifyResult', array('success' => ($success) ? 'true' : 'false', 'msg' => $msg)); $this->endXML(); } }