3 * Laconica, the distributed open-source microblogging tool
5 * Base class for RSS 1.0 feed actions
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 Evan Prodromou <evan@controlyourself.ca>
25 * @author Earle Martin <earle@downlode.org>
26 * @copyright 2008-9 Control Yourself, Inc.
27 * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
28 * @link http://laconi.ca/
31 if (!defined('LACONICA')) { exit(1); }
33 define('DEFAULT_RSS_LIMIT', 48);
35 class Rss10Action extends Action
37 # This will contain the details of each feed item's author and be used to generate SIOC data.
39 var $creators = array();
40 var $limit = DEFAULT_RSS_LIMIT;
42 var $tags_already_output = array();
47 * Just wraps the Action constructor.
49 * @param string $output URI to output to, default = stdout
50 * @param boolean $indent Whether to indent output, default true
52 * @see Action::__construct
55 function __construct($output='php://output', $indent=true)
57 parent::__construct($output, $indent);
61 * Do we need to write to the database?
63 * @return boolean true
72 * Read arguments and initialize members
74 * @param array $args Arguments from $_REQUEST
75 * @return boolean success
78 function prepare($args)
80 parent::prepare($args);
81 $this->limit = (int) $this->trimmed('limit');
82 if ($this->limit == 0) {
83 $this->limit = DEFAULT_RSS_LIMIT;
91 * @param array $args Arguments from $_REQUEST
96 function handle($args)
98 // Parent handling, including cache check
99 parent::handle($args);
101 if (common_config('site', 'private')) {
102 if (!isset($_SERVER['PHP_AUTH_USER'])) {
104 # This header makes basic auth go
105 header('WWW-Authenticate: Basic realm="Laconica RSS"');
107 # If the user hits cancel -- bam!
108 $this->show_basic_auth_error();
111 $nickname = $_SERVER['PHP_AUTH_USER'];
112 $password = $_SERVER['PHP_AUTH_PW'];
114 if (!common_check_user($nickname, $password)) {
115 # basic authentication failed
116 list($proxy, $ip) = common_client_ip();
118 common_log(LOG_WARNING, "Failed RSS auth attempt, nickname = $nickname, proxy = $proxy, ip = $ip.");
119 $this->show_basic_auth_error();
125 // Get the list of notices
126 if (empty($this->tag)) {
127 $this->notices = $this->getNotices($this->limit);
129 $this->notices = $this->getTaggedNotices($this->tag, $this->limit);
134 function show_basic_auth_error()
136 header('HTTP/1.1 401 Unauthorized');
137 header('Content-Type: application/xml; charset=utf-8');
139 $this->elementStart('hash');
140 $this->element('error', null, 'Could not authenticate you.');
141 $this->element('request', null, $_SERVER['REQUEST_URI']);
142 $this->elementEnd('hash');
147 * Get the notices to output in this stream
149 * @return array an array of Notice objects sorted in reverse chron
152 function getNotices()
158 * Get a description of the channel
160 * Returns an array with the following
164 function getChannel()
166 return array('url' => '',
169 'description' => '');
180 $this->showChannel();
183 foreach ($this->notices as $n) {
187 $this->showCreators();
191 function showChannel()
194 $channel = $this->getChannel();
195 $image = $this->getImage();
197 $this->elementStart('channel', array('rdf:about' => $channel['url']));
198 $this->element('title', null, $channel['title']);
199 $this->element('link', null, $channel['link']);
200 $this->element('description', null, $channel['description']);
201 $this->element('cc:licence', array('rdf:resource' => common_config('license','url')));
204 $this->element('image', array('rdf:resource' => $image));
207 $this->elementStart('items');
208 $this->elementStart('rdf:Seq');
210 foreach ($this->notices as $notice) {
211 $this->element('rdf:li', array('rdf:resource' => $notice->uri));
214 $this->elementEnd('rdf:Seq');
215 $this->elementEnd('items');
217 $this->elementEnd('channel');
222 $image = $this->getImage();
224 $channel = $this->getChannel();
225 $this->elementStart('image', array('rdf:about' => $image));
226 $this->element('title', null, $channel['title']);
227 $this->element('link', null, $channel['link']);
228 $this->element('url', null, $image);
229 $this->elementEnd('image');
233 function showItem($notice)
235 $profile = Profile::staticGet($notice->profile_id);
236 $nurl = common_local_url('shownotice', array('notice' => $notice->id));
237 $creator_uri = common_profile_uri($profile);
238 $this->elementStart('item', array('rdf:about' => $notice->uri,
239 'rdf:type' => 'http://rdfs.org/sioc/types#MicroblogPost'));
240 $title = $profile->nickname . ': ' . common_xml_safe_str(trim($notice->content));
241 $this->element('title', null, $title);
242 $this->element('link', null, $nurl);
243 $this->element('description', null, $profile->nickname."'s status on ".common_exact_date($notice->created));
244 if ($notice->rendered) {
245 $this->element('content:encoded', null, common_xml_safe_str($notice->rendered));
247 $this->element('dc:date', null, common_date_w3dtf($notice->created));
248 $this->element('dc:creator', null, ($profile->fullname) ? $profile->fullname : $profile->nickname);
249 $this->element('foaf:maker', array('rdf:resource' => $creator_uri));
250 $this->element('sioc:has_creator', array('rdf:resource' => $creator_uri.'#acct'));
251 $this->element('laconica:postIcon', array('rdf:resource' => $profile->avatarUrl()));
252 $this->element('cc:licence', array('rdf:resource' => common_config('license', 'url')));
253 if ($notice->reply_to) {
254 $replyurl = common_local_url('shownotice', array('notice' => $notice->reply_to));
255 $this->element('sioc:reply_of', array('rdf:resource' => $replyurl));
257 if (!empty($notice->conversation)) {
258 $conversationurl = common_local_url('conversation',
259 array('id' => $notice->conversation));
260 $this->element('sioc:has_discussion', array('rdf:resource' => $conversationurl));
262 $attachments = $notice->attachments();
264 foreach($attachments as $attachment){
265 if ($attachment->isEnclosure()) {
266 // DO NOT move xmlns declaration to root element. Making it
267 // the default namespace here improves compatibility with
268 // real-world feed readers.
270 'rdf:resource' => $attachment->url,
271 'url' => $attachment->url,
272 'xmlns' => 'http://purl.oclc.org/net/rss_2.0/enc#'
274 if ($attachment->title) {
275 $attribs['dc:title'] = $attachment->title;
277 if ($attachment->modified) {
278 $attribs['dc:date'] = common_date_w3dtf($attachment->modified);
280 if ($attachment->size) {
281 $attribs['length'] = $attachment->size;
283 if ($attachment->mimetype) {
284 $attribs['type'] = $attachment->mimetype;
286 $this->element('enclosure', $attribs);
288 $this->element('sioc:links_to', array('rdf:resource'=>$attachment->url));
292 $tag = new Notice_tag();
293 $tag->notice_id = $notice->id;
295 $entry['tags']=array();
296 while ($tag->fetch()) {
297 $tagpage = common_local_url('tag', array('tag' => $tag->tag));
299 if ( in_array($tag, $this->tags_already_output) ) {
300 $this->element('ctag:tagged', array('rdf:resource'=>$tagpage.'#concept'));
304 $tagrss = common_local_url('tagrss', array('tag' => $tag->tag));
305 $this->elementStart('ctag:tagged');
306 $this->elementStart('ctag:Tag', array('rdf:about'=>$tagpage.'#concept', 'ctag:label'=>$tag->tag));
307 $this->element('foaf:page', array('rdf:resource'=>$tagpage));
308 $this->element('rdfs:seeAlso', array('rdf:resource'=>$tagrss));
309 $this->elementEnd('ctag:Tag');
310 $this->elementEnd('ctag:tagged');
312 $this->tags_already_output[] = $tag->tag;
315 $this->elementEnd('item');
316 $this->creators[$creator_uri] = $profile;
319 function showCreators()
321 foreach ($this->creators as $uri => $profile) {
323 $nickname = $profile->nickname;
324 $this->elementStart('foaf:Agent', array('rdf:about' => $uri));
325 $this->element('foaf:nick', null, $nickname);
326 if ($profile->fullname) {
327 $this->element('foaf:name', null, $profile->fullname);
329 $this->element('foaf:holdsAccount', array('rdf:resource' => $uri.'#acct'));
330 $avatar = $profile->avatarUrl();
331 $this->element('foaf:depiction', array('rdf:resource' => $avatar));
332 $this->elementEnd('foaf:Agent');
338 $channel = $this->getChannel();
339 header('Content-Type: application/rdf+xml');
342 $this->elementStart('rdf:RDF', array('xmlns:rdf' =>
343 'http://www.w3.org/1999/02/22-rdf-syntax-ns#',
345 'http://purl.org/dc/elements/1.1/',
347 'http://creativecommons.org/ns#',
349 'http://purl.org/rss/1.0/modules/content/',
351 'http://commontag.org/ns#',
353 'http://xmlns.com/foaf/0.1/',
355 'http://rdfs.org/sioc/ns#',
357 'http://rdfs.org/sioc/types#',
359 'http://www.w3.org/2000/01/rdf-schema#',
361 'http://laconi.ca/ont/',
362 'xmlns' => 'http://purl.org/rss/1.0/'));
363 $this->elementStart('sioc:Site', array('rdf:about' => common_root_url()));
364 $this->element('sioc:name', null, common_config('site', 'name'));
365 $this->elementStart('sioc:space_of');
366 $this->element('sioc:Container', array('rdf:about' =>
368 $this->elementEnd('sioc:space_of');
369 $this->elementEnd('sioc:Site');
374 $this->elementEnd('rdf:RDF');
378 * When was this page last modified?
382 function lastModified()
384 if (empty($this->notices)) {
388 if (count($this->notices) == 0) {
392 // FIXME: doesn't handle modified profiles, avatars, deleted notices
394 return strtotime($this->notices[0]->created);