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;
46 * Just wraps the Action constructor.
48 * @param string $output URI to output to, default = stdout
49 * @param boolean $indent Whether to indent output, default true
51 * @see Action::__construct
54 function __construct($output='php://output', $indent=true)
56 parent::__construct($output, $indent);
60 * Do we need to write to the database?
62 * @return boolean true
71 * Read arguments and initialize members
73 * @param array $args Arguments from $_REQUEST
74 * @return boolean success
77 function prepare($args)
79 parent::prepare($args);
80 $this->limit = (int) $this->trimmed('limit');
81 if ($this->limit == 0) {
82 $this->limit = DEFAULT_RSS_LIMIT;
90 * @param array $args Arguments from $_REQUEST
95 function handle($args)
97 // Parent handling, including cache check
98 parent::handle($args);
99 // Get the list of notices
100 $this->notices = $this->getNotices($this->limit);
105 * Get the notices to output in this stream
107 * @return array an array of Notice objects sorted in reverse chron
110 function getNotices()
116 * Get a description of the channel
118 * Returns an array with the following
122 function getChannel()
124 return array('url' => '',
127 'description' => '');
138 $this->showChannel();
141 foreach ($this->notices as $n) {
145 $this->showCreators();
149 function showChannel()
152 $channel = $this->getChannel();
153 $image = $this->getImage();
155 $this->elementStart('channel', array('rdf:about' => $channel['url']));
156 $this->element('title', null, $channel['title']);
157 $this->element('link', null, $channel['link']);
158 $this->element('description', null, $channel['description']);
159 $this->element('cc:licence', array('rdf:resource' => common_config('license','url')));
162 $this->element('image', array('rdf:resource' => $image));
165 $this->elementStart('items');
166 $this->elementStart('rdf:Seq');
168 foreach ($this->notices as $notice) {
169 $this->element('sioct:MicroblogPost', array('rdf:resource' => $notice->uri));
172 $this->elementEnd('rdf:Seq');
173 $this->elementEnd('items');
175 $this->elementEnd('channel');
180 $image = $this->getImage();
182 $channel = $this->getChannel();
183 $this->elementStart('image', array('rdf:about' => $image));
184 $this->element('title', null, $channel['title']);
185 $this->element('link', null, $channel['link']);
186 $this->element('url', null, $image);
187 $this->elementEnd('image');
191 function showItem($notice)
193 $profile = Profile::staticGet($notice->profile_id);
194 $nurl = common_local_url('shownotice', array('notice' => $notice->id));
195 $creator_uri = common_profile_uri($profile);
196 $this->elementStart('item', array('rdf:about' => $notice->uri));
197 $title = $profile->nickname . ': ' . common_xml_safe_str(trim($notice->content));
198 $this->element('title', null, $title);
199 $this->element('link', null, $nurl);
200 $this->element('description', null, $profile->nickname."'s status on ".common_exact_date($notice->created));
201 $this->element('dc:date', null, common_date_w3dtf($notice->created));
202 $this->element('dc:creator', null, ($profile->fullname) ? $profile->fullname : $profile->nickname);
203 $this->element('sioc:has_creator', array('rdf:resource' => $creator_uri));
204 $this->element('laconica:postIcon', array('rdf:resource' => $profile->avatarUrl()));
205 $this->element('cc:licence', array('rdf:resource' => common_config('license', 'url')));
206 $this->elementEnd('item');
207 $this->creators[$creator_uri] = $profile;
210 function showCreators()
212 foreach ($this->creators as $uri => $profile) {
214 $nickname = $profile->nickname;
215 $this->elementStart('sioc:User', array('rdf:about' => $uri));
216 $this->element('foaf:nick', null, $nickname);
217 if ($profile->fullname) {
218 $this->element('foaf:name', null, $profile->fullname);
220 $this->element('sioc:id', null, $id);
221 $avatar = $profile->avatarUrl();
222 $this->element('sioc:avatar', array('rdf:resource' => $avatar));
223 $this->elementEnd('sioc:User');
229 $channel = $this->getChannel();
230 header('Content-Type: application/rdf+xml');
233 $this->elementStart('rdf:RDF', array('xmlns:rdf' =>
234 'http://www.w3.org/1999/02/22-rdf-syntax-ns#',
236 'http://purl.org/dc/elements/1.1/',
238 'http://web.resource.org/cc/',
240 'http://purl.org/rss/1.0/modules/content/',
242 'http://xmlns.com/foaf/0.1/',
244 'http://rdfs.org/sioc/ns#',
246 'http://rdfs.org/sioc/types#',
248 'http://laconi.ca/ont/',
249 'xmlns' => 'http://purl.org/rss/1.0/'));
250 $this->elementStart('sioc:Site', array('rdf:about' => common_root_url()));
251 $this->element('sioc:name', null, common_config('site', 'name'));
252 $this->elementStart('sioc:container_of');
253 $this->element('sioc:Container', array('rdf:about' =>
255 $this->elementEnd('sioc:container_of');
256 $this->elementEnd('sioc:Site');
261 $this->elementEnd('rdf:RDF');
265 * When was this page last modified?
269 function lastModified()
271 if (empty($this->notices)) {
275 if (count($this->notices) == 0) {
279 // FIXME: doesn't handle modified profiles, avatars, deleted notices
281 return strtotime($this->notices[0]->created);