author | ymh <ymh.work@gmail.com> |
Tue, 27 Sep 2022 16:37:53 +0200 | |
changeset 19 | 3d72ae0968f4 |
parent 18 | be944660c56a |
child 21 | 48c4eec2b7e6 |
permissions | -rw-r--r-- |
0 | 1 |
<?php |
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
2 |
if ( ! class_exists( 'SimplePie', false ) ) : |
0 | 3 |
|
4 |
// Load classes we will need. |
|
5 |
require ABSPATH . WPINC . '/SimplePie/Misc.php'; |
|
6 |
require ABSPATH . WPINC . '/SimplePie/Cache.php'; |
|
7 |
require ABSPATH . WPINC . '/SimplePie/File.php'; |
|
8 |
require ABSPATH . WPINC . '/SimplePie/Sanitize.php'; |
|
9 |
require ABSPATH . WPINC . '/SimplePie/Registry.php'; |
|
10 |
require ABSPATH . WPINC . '/SimplePie/IRI.php'; |
|
11 |
require ABSPATH . WPINC . '/SimplePie/Locator.php'; |
|
12 |
require ABSPATH . WPINC . '/SimplePie/Content/Type/Sniffer.php'; |
|
13 |
require ABSPATH . WPINC . '/SimplePie/XML/Declaration/Parser.php'; |
|
14 |
require ABSPATH . WPINC . '/SimplePie/Parser.php'; |
|
15 |
require ABSPATH . WPINC . '/SimplePie/Item.php'; |
|
16 |
require ABSPATH . WPINC . '/SimplePie/Parse/Date.php'; |
|
17 |
require ABSPATH . WPINC . '/SimplePie/Author.php'; |
|
18 |
||
19 |
/** |
|
20 |
* WordPress autoloader for SimplePie. |
|
21 |
* |
|
22 |
* @since 3.5.0 |
|
23 |
*/ |
|
24 |
function wp_simplepie_autoload( $class ) { |
|
25 |
if ( 0 !== strpos( $class, 'SimplePie_' ) ) |
|
26 |
return; |
|
27 |
||
28 |
$file = ABSPATH . WPINC . '/' . str_replace( '_', '/', $class ) . '.php'; |
|
16 | 29 |
include $file; |
0 | 30 |
} |
31 |
||
7
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
32 |
/** |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
33 |
* We autoload classes we may not need. |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
34 |
*/ |
cf61fcea0001
resynchronize code repo with production
ymh <ymh.work@gmail.com>
parents:
5
diff
changeset
|
35 |
spl_autoload_register( 'wp_simplepie_autoload' ); |
0 | 36 |
|
37 |
/** |
|
38 |
* SimplePie |
|
39 |
* |
|
40 |
* A PHP-Based RSS and Atom Feed Framework. |
|
41 |
* Takes the hard work out of managing a complete RSS/Atom solution. |
|
42 |
* |
|
16 | 43 |
* Copyright (c) 2004-2017, Ryan Parman, Sam Sneddon, Ryan McCue, and contributors |
0 | 44 |
* All rights reserved. |
45 |
* |
|
46 |
* Redistribution and use in source and binary forms, with or without modification, are |
|
47 |
* permitted provided that the following conditions are met: |
|
48 |
* |
|
49 |
* * Redistributions of source code must retain the above copyright notice, this list of |
|
50 |
* conditions and the following disclaimer. |
|
51 |
* |
|
52 |
* * Redistributions in binary form must reproduce the above copyright notice, this list |
|
53 |
* of conditions and the following disclaimer in the documentation and/or other materials |
|
54 |
* provided with the distribution. |
|
55 |
* |
|
56 |
* * Neither the name of the SimplePie Team nor the names of its contributors may be used |
|
57 |
* to endorse or promote products derived from this software without specific prior |
|
58 |
* written permission. |
|
59 |
* |
|
60 |
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS |
|
61 |
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY |
|
62 |
* AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS |
|
63 |
* AND CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
|
64 |
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
|
65 |
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
|
66 |
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR |
|
67 |
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
|
68 |
* POSSIBILITY OF SUCH DAMAGE. |
|
69 |
* |
|
70 |
* @package SimplePie |
|
19 | 71 |
* @version 1.5.8 |
16 | 72 |
* @copyright 2004-2017 Ryan Parman, Sam Sneddon, Ryan McCue |
0 | 73 |
* @author Ryan Parman |
16 | 74 |
* @author Sam Sneddon |
0 | 75 |
* @author Ryan McCue |
76 |
* @link http://simplepie.org/ SimplePie |
|
77 |
* @license http://www.opensource.org/licenses/bsd-license.php BSD License |
|
78 |
*/ |
|
79 |
||
80 |
/** |
|
81 |
* SimplePie Name |
|
82 |
*/ |
|
83 |
define('SIMPLEPIE_NAME', 'SimplePie'); |
|
84 |
||
85 |
/** |
|
86 |
* SimplePie Version |
|
87 |
*/ |
|
19 | 88 |
define('SIMPLEPIE_VERSION', '1.5.8'); |
0 | 89 |
|
90 |
/** |
|
91 |
* SimplePie Build |
|
92 |
* @todo Hardcode for release (there's no need to have to call SimplePie_Misc::get_build() only every load of simplepie.inc) |
|
93 |
*/ |
|
94 |
define('SIMPLEPIE_BUILD', gmdate('YmdHis', SimplePie_Misc::get_build())); |
|
95 |
||
96 |
/** |
|
97 |
* SimplePie Website URL |
|
98 |
*/ |
|
99 |
define('SIMPLEPIE_URL', 'http://simplepie.org'); |
|
100 |
||
101 |
/** |
|
102 |
* SimplePie Useragent |
|
103 |
* @see SimplePie::set_useragent() |
|
104 |
*/ |
|
105 |
define('SIMPLEPIE_USERAGENT', SIMPLEPIE_NAME . '/' . SIMPLEPIE_VERSION . ' (Feed Parser; ' . SIMPLEPIE_URL . '; Allow like Gecko) Build/' . SIMPLEPIE_BUILD); |
|
106 |
||
107 |
/** |
|
108 |
* SimplePie Linkback |
|
109 |
*/ |
|
110 |
define('SIMPLEPIE_LINKBACK', '<a href="' . SIMPLEPIE_URL . '" title="' . SIMPLEPIE_NAME . ' ' . SIMPLEPIE_VERSION . '">' . SIMPLEPIE_NAME . '</a>'); |
|
111 |
||
112 |
/** |
|
113 |
* No Autodiscovery |
|
114 |
* @see SimplePie::set_autodiscovery_level() |
|
115 |
*/ |
|
116 |
define('SIMPLEPIE_LOCATOR_NONE', 0); |
|
117 |
||
118 |
/** |
|
119 |
* Feed Link Element Autodiscovery |
|
120 |
* @see SimplePie::set_autodiscovery_level() |
|
121 |
*/ |
|
122 |
define('SIMPLEPIE_LOCATOR_AUTODISCOVERY', 1); |
|
123 |
||
124 |
/** |
|
125 |
* Local Feed Extension Autodiscovery |
|
126 |
* @see SimplePie::set_autodiscovery_level() |
|
127 |
*/ |
|
128 |
define('SIMPLEPIE_LOCATOR_LOCAL_EXTENSION', 2); |
|
129 |
||
130 |
/** |
|
131 |
* Local Feed Body Autodiscovery |
|
132 |
* @see SimplePie::set_autodiscovery_level() |
|
133 |
*/ |
|
134 |
define('SIMPLEPIE_LOCATOR_LOCAL_BODY', 4); |
|
135 |
||
136 |
/** |
|
137 |
* Remote Feed Extension Autodiscovery |
|
138 |
* @see SimplePie::set_autodiscovery_level() |
|
139 |
*/ |
|
140 |
define('SIMPLEPIE_LOCATOR_REMOTE_EXTENSION', 8); |
|
141 |
||
142 |
/** |
|
143 |
* Remote Feed Body Autodiscovery |
|
144 |
* @see SimplePie::set_autodiscovery_level() |
|
145 |
*/ |
|
146 |
define('SIMPLEPIE_LOCATOR_REMOTE_BODY', 16); |
|
147 |
||
148 |
/** |
|
149 |
* All Feed Autodiscovery |
|
150 |
* @see SimplePie::set_autodiscovery_level() |
|
151 |
*/ |
|
152 |
define('SIMPLEPIE_LOCATOR_ALL', 31); |
|
153 |
||
154 |
/** |
|
155 |
* No known feed type |
|
156 |
*/ |
|
157 |
define('SIMPLEPIE_TYPE_NONE', 0); |
|
158 |
||
159 |
/** |
|
160 |
* RSS 0.90 |
|
161 |
*/ |
|
162 |
define('SIMPLEPIE_TYPE_RSS_090', 1); |
|
163 |
||
164 |
/** |
|
165 |
* RSS 0.91 (Netscape) |
|
166 |
*/ |
|
167 |
define('SIMPLEPIE_TYPE_RSS_091_NETSCAPE', 2); |
|
168 |
||
169 |
/** |
|
170 |
* RSS 0.91 (Userland) |
|
171 |
*/ |
|
172 |
define('SIMPLEPIE_TYPE_RSS_091_USERLAND', 4); |
|
173 |
||
174 |
/** |
|
175 |
* RSS 0.91 (both Netscape and Userland) |
|
176 |
*/ |
|
177 |
define('SIMPLEPIE_TYPE_RSS_091', 6); |
|
178 |
||
179 |
/** |
|
180 |
* RSS 0.92 |
|
181 |
*/ |
|
182 |
define('SIMPLEPIE_TYPE_RSS_092', 8); |
|
183 |
||
184 |
/** |
|
185 |
* RSS 0.93 |
|
186 |
*/ |
|
187 |
define('SIMPLEPIE_TYPE_RSS_093', 16); |
|
188 |
||
189 |
/** |
|
190 |
* RSS 0.94 |
|
191 |
*/ |
|
192 |
define('SIMPLEPIE_TYPE_RSS_094', 32); |
|
193 |
||
194 |
/** |
|
195 |
* RSS 1.0 |
|
196 |
*/ |
|
197 |
define('SIMPLEPIE_TYPE_RSS_10', 64); |
|
198 |
||
199 |
/** |
|
200 |
* RSS 2.0 |
|
201 |
*/ |
|
202 |
define('SIMPLEPIE_TYPE_RSS_20', 128); |
|
203 |
||
204 |
/** |
|
205 |
* RDF-based RSS |
|
206 |
*/ |
|
207 |
define('SIMPLEPIE_TYPE_RSS_RDF', 65); |
|
208 |
||
209 |
/** |
|
210 |
* Non-RDF-based RSS (truly intended as syndication format) |
|
211 |
*/ |
|
212 |
define('SIMPLEPIE_TYPE_RSS_SYNDICATION', 190); |
|
213 |
||
214 |
/** |
|
215 |
* All RSS |
|
216 |
*/ |
|
217 |
define('SIMPLEPIE_TYPE_RSS_ALL', 255); |
|
218 |
||
219 |
/** |
|
220 |
* Atom 0.3 |
|
221 |
*/ |
|
222 |
define('SIMPLEPIE_TYPE_ATOM_03', 256); |
|
223 |
||
224 |
/** |
|
225 |
* Atom 1.0 |
|
226 |
*/ |
|
227 |
define('SIMPLEPIE_TYPE_ATOM_10', 512); |
|
228 |
||
229 |
/** |
|
230 |
* All Atom |
|
231 |
*/ |
|
232 |
define('SIMPLEPIE_TYPE_ATOM_ALL', 768); |
|
233 |
||
234 |
/** |
|
235 |
* All feed types |
|
236 |
*/ |
|
237 |
define('SIMPLEPIE_TYPE_ALL', 1023); |
|
238 |
||
239 |
/** |
|
240 |
* No construct |
|
241 |
*/ |
|
242 |
define('SIMPLEPIE_CONSTRUCT_NONE', 0); |
|
243 |
||
244 |
/** |
|
245 |
* Text construct |
|
246 |
*/ |
|
247 |
define('SIMPLEPIE_CONSTRUCT_TEXT', 1); |
|
248 |
||
249 |
/** |
|
250 |
* HTML construct |
|
251 |
*/ |
|
252 |
define('SIMPLEPIE_CONSTRUCT_HTML', 2); |
|
253 |
||
254 |
/** |
|
255 |
* XHTML construct |
|
256 |
*/ |
|
257 |
define('SIMPLEPIE_CONSTRUCT_XHTML', 4); |
|
258 |
||
259 |
/** |
|
260 |
* base64-encoded construct |
|
261 |
*/ |
|
262 |
define('SIMPLEPIE_CONSTRUCT_BASE64', 8); |
|
263 |
||
264 |
/** |
|
265 |
* IRI construct |
|
266 |
*/ |
|
267 |
define('SIMPLEPIE_CONSTRUCT_IRI', 16); |
|
268 |
||
269 |
/** |
|
270 |
* A construct that might be HTML |
|
271 |
*/ |
|
272 |
define('SIMPLEPIE_CONSTRUCT_MAYBE_HTML', 32); |
|
273 |
||
274 |
/** |
|
275 |
* All constructs |
|
276 |
*/ |
|
277 |
define('SIMPLEPIE_CONSTRUCT_ALL', 63); |
|
278 |
||
279 |
/** |
|
280 |
* Don't change case |
|
281 |
*/ |
|
282 |
define('SIMPLEPIE_SAME_CASE', 1); |
|
283 |
||
284 |
/** |
|
285 |
* Change to lowercase |
|
286 |
*/ |
|
287 |
define('SIMPLEPIE_LOWERCASE', 2); |
|
288 |
||
289 |
/** |
|
290 |
* Change to uppercase |
|
291 |
*/ |
|
292 |
define('SIMPLEPIE_UPPERCASE', 4); |
|
293 |
||
294 |
/** |
|
295 |
* PCRE for HTML attributes |
|
296 |
*/ |
|
297 |
define('SIMPLEPIE_PCRE_HTML_ATTRIBUTE', '((?:[\x09\x0A\x0B\x0C\x0D\x20]+[^\x09\x0A\x0B\x0C\x0D\x20\x2F\x3E][^\x09\x0A\x0B\x0C\x0D\x20\x2F\x3D\x3E]*(?:[\x09\x0A\x0B\x0C\x0D\x20]*=[\x09\x0A\x0B\x0C\x0D\x20]*(?:"(?:[^"]*)"|\'(?:[^\']*)\'|(?:[^\x09\x0A\x0B\x0C\x0D\x20\x22\x27\x3E][^\x09\x0A\x0B\x0C\x0D\x20\x3E]*)?))?)*)[\x09\x0A\x0B\x0C\x0D\x20]*'); |
|
298 |
||
299 |
/** |
|
300 |
* PCRE for XML attributes |
|
301 |
*/ |
|
302 |
define('SIMPLEPIE_PCRE_XML_ATTRIBUTE', '((?:\s+(?:(?:[^\s:]+:)?[^\s:]+)\s*=\s*(?:"(?:[^"]*)"|\'(?:[^\']*)\'))*)\s*'); |
|
303 |
||
304 |
/** |
|
305 |
* XML Namespace |
|
306 |
*/ |
|
307 |
define('SIMPLEPIE_NAMESPACE_XML', 'http://www.w3.org/XML/1998/namespace'); |
|
308 |
||
309 |
/** |
|
310 |
* Atom 1.0 Namespace |
|
311 |
*/ |
|
312 |
define('SIMPLEPIE_NAMESPACE_ATOM_10', 'http://www.w3.org/2005/Atom'); |
|
313 |
||
314 |
/** |
|
315 |
* Atom 0.3 Namespace |
|
316 |
*/ |
|
317 |
define('SIMPLEPIE_NAMESPACE_ATOM_03', 'http://purl.org/atom/ns#'); |
|
318 |
||
319 |
/** |
|
320 |
* RDF Namespace |
|
321 |
*/ |
|
322 |
define('SIMPLEPIE_NAMESPACE_RDF', 'http://www.w3.org/1999/02/22-rdf-syntax-ns#'); |
|
323 |
||
324 |
/** |
|
325 |
* RSS 0.90 Namespace |
|
326 |
*/ |
|
327 |
define('SIMPLEPIE_NAMESPACE_RSS_090', 'http://my.netscape.com/rdf/simple/0.9/'); |
|
328 |
||
329 |
/** |
|
330 |
* RSS 1.0 Namespace |
|
331 |
*/ |
|
332 |
define('SIMPLEPIE_NAMESPACE_RSS_10', 'http://purl.org/rss/1.0/'); |
|
333 |
||
334 |
/** |
|
335 |
* RSS 1.0 Content Module Namespace |
|
336 |
*/ |
|
337 |
define('SIMPLEPIE_NAMESPACE_RSS_10_MODULES_CONTENT', 'http://purl.org/rss/1.0/modules/content/'); |
|
338 |
||
339 |
/** |
|
340 |
* RSS 2.0 Namespace |
|
341 |
* (Stupid, I know, but I'm certain it will confuse people less with support.) |
|
342 |
*/ |
|
343 |
define('SIMPLEPIE_NAMESPACE_RSS_20', ''); |
|
344 |
||
345 |
/** |
|
346 |
* DC 1.0 Namespace |
|
347 |
*/ |
|
348 |
define('SIMPLEPIE_NAMESPACE_DC_10', 'http://purl.org/dc/elements/1.0/'); |
|
349 |
||
350 |
/** |
|
351 |
* DC 1.1 Namespace |
|
352 |
*/ |
|
353 |
define('SIMPLEPIE_NAMESPACE_DC_11', 'http://purl.org/dc/elements/1.1/'); |
|
354 |
||
355 |
/** |
|
356 |
* W3C Basic Geo (WGS84 lat/long) Vocabulary Namespace |
|
357 |
*/ |
|
358 |
define('SIMPLEPIE_NAMESPACE_W3C_BASIC_GEO', 'http://www.w3.org/2003/01/geo/wgs84_pos#'); |
|
359 |
||
360 |
/** |
|
361 |
* GeoRSS Namespace |
|
362 |
*/ |
|
363 |
define('SIMPLEPIE_NAMESPACE_GEORSS', 'http://www.georss.org/georss'); |
|
364 |
||
365 |
/** |
|
366 |
* Media RSS Namespace |
|
367 |
*/ |
|
368 |
define('SIMPLEPIE_NAMESPACE_MEDIARSS', 'http://search.yahoo.com/mrss/'); |
|
369 |
||
370 |
/** |
|
371 |
* Wrong Media RSS Namespace. Caused by a long-standing typo in the spec. |
|
372 |
*/ |
|
373 |
define('SIMPLEPIE_NAMESPACE_MEDIARSS_WRONG', 'http://search.yahoo.com/mrss'); |
|
374 |
||
375 |
/** |
|
376 |
* Wrong Media RSS Namespace #2. New namespace introduced in Media RSS 1.5. |
|
377 |
*/ |
|
378 |
define('SIMPLEPIE_NAMESPACE_MEDIARSS_WRONG2', 'http://video.search.yahoo.com/mrss'); |
|
379 |
||
380 |
/** |
|
381 |
* Wrong Media RSS Namespace #3. A possible typo of the Media RSS 1.5 namespace. |
|
382 |
*/ |
|
383 |
define('SIMPLEPIE_NAMESPACE_MEDIARSS_WRONG3', 'http://video.search.yahoo.com/mrss/'); |
|
384 |
||
385 |
/** |
|
386 |
* Wrong Media RSS Namespace #4. New spec location after the RSS Advisory Board takes it over, but not a valid namespace. |
|
387 |
*/ |
|
388 |
define('SIMPLEPIE_NAMESPACE_MEDIARSS_WRONG4', 'http://www.rssboard.org/media-rss'); |
|
389 |
||
390 |
/** |
|
391 |
* Wrong Media RSS Namespace #5. A possible typo of the RSS Advisory Board URL. |
|
392 |
*/ |
|
393 |
define('SIMPLEPIE_NAMESPACE_MEDIARSS_WRONG5', 'http://www.rssboard.org/media-rss/'); |
|
394 |
||
395 |
/** |
|
396 |
* iTunes RSS Namespace |
|
397 |
*/ |
|
398 |
define('SIMPLEPIE_NAMESPACE_ITUNES', 'http://www.itunes.com/dtds/podcast-1.0.dtd'); |
|
399 |
||
400 |
/** |
|
401 |
* XHTML Namespace |
|
402 |
*/ |
|
403 |
define('SIMPLEPIE_NAMESPACE_XHTML', 'http://www.w3.org/1999/xhtml'); |
|
404 |
||
405 |
/** |
|
406 |
* IANA Link Relations Registry |
|
407 |
*/ |
|
408 |
define('SIMPLEPIE_IANA_LINK_RELATIONS_REGISTRY', 'http://www.iana.org/assignments/relation/'); |
|
409 |
||
410 |
/** |
|
411 |
* No file source |
|
412 |
*/ |
|
413 |
define('SIMPLEPIE_FILE_SOURCE_NONE', 0); |
|
414 |
||
415 |
/** |
|
416 |
* Remote file source |
|
417 |
*/ |
|
418 |
define('SIMPLEPIE_FILE_SOURCE_REMOTE', 1); |
|
419 |
||
420 |
/** |
|
421 |
* Local file source |
|
422 |
*/ |
|
423 |
define('SIMPLEPIE_FILE_SOURCE_LOCAL', 2); |
|
424 |
||
425 |
/** |
|
426 |
* fsockopen() file source |
|
427 |
*/ |
|
428 |
define('SIMPLEPIE_FILE_SOURCE_FSOCKOPEN', 4); |
|
429 |
||
430 |
/** |
|
431 |
* cURL file source |
|
432 |
*/ |
|
433 |
define('SIMPLEPIE_FILE_SOURCE_CURL', 8); |
|
434 |
||
435 |
/** |
|
436 |
* file_get_contents() file source |
|
437 |
*/ |
|
438 |
define('SIMPLEPIE_FILE_SOURCE_FILE_GET_CONTENTS', 16); |
|
439 |
||
440 |
||
441 |
||
442 |
/** |
|
443 |
* SimplePie |
|
444 |
* |
|
445 |
* @package SimplePie |
|
446 |
* @subpackage API |
|
447 |
*/ |
|
448 |
class SimplePie |
|
449 |
{ |
|
450 |
/** |
|
451 |
* @var array Raw data |
|
452 |
* @access private |
|
453 |
*/ |
|
454 |
public $data = array(); |
|
455 |
||
456 |
/** |
|
457 |
* @var mixed Error string |
|
458 |
* @access private |
|
459 |
*/ |
|
460 |
public $error; |
|
461 |
||
462 |
/** |
|
19 | 463 |
* @var int HTTP status code |
464 |
* @see SimplePie::status_code() |
|
465 |
* @access private |
|
466 |
*/ |
|
467 |
public $status_code; |
|
468 |
||
469 |
/** |
|
0 | 470 |
* @var object Instance of SimplePie_Sanitize (or other class) |
471 |
* @see SimplePie::set_sanitize_class() |
|
472 |
* @access private |
|
473 |
*/ |
|
474 |
public $sanitize; |
|
475 |
||
476 |
/** |
|
477 |
* @var string SimplePie Useragent |
|
478 |
* @see SimplePie::set_useragent() |
|
479 |
* @access private |
|
480 |
*/ |
|
481 |
public $useragent = SIMPLEPIE_USERAGENT; |
|
482 |
||
483 |
/** |
|
484 |
* @var string Feed URL |
|
485 |
* @see SimplePie::set_feed_url() |
|
486 |
* @access private |
|
487 |
*/ |
|
488 |
public $feed_url; |
|
489 |
||
490 |
/** |
|
16 | 491 |
* @var string Original feed URL, or new feed URL iff HTTP 301 Moved Permanently |
492 |
* @see SimplePie::subscribe_url() |
|
493 |
* @access private |
|
494 |
*/ |
|
495 |
public $permanent_url = null; |
|
496 |
||
497 |
/** |
|
0 | 498 |
* @var object Instance of SimplePie_File to use as a feed |
499 |
* @see SimplePie::set_file() |
|
500 |
* @access private |
|
501 |
*/ |
|
502 |
public $file; |
|
503 |
||
504 |
/** |
|
505 |
* @var string Raw feed data |
|
506 |
* @see SimplePie::set_raw_data() |
|
507 |
* @access private |
|
508 |
*/ |
|
509 |
public $raw_data; |
|
510 |
||
511 |
/** |
|
512 |
* @var int Timeout for fetching remote files |
|
513 |
* @see SimplePie::set_timeout() |
|
514 |
* @access private |
|
515 |
*/ |
|
516 |
public $timeout = 10; |
|
517 |
||
518 |
/** |
|
16 | 519 |
* @var array Custom curl options |
520 |
* @see SimplePie::set_curl_options() |
|
521 |
* @access private |
|
522 |
*/ |
|
523 |
public $curl_options = array(); |
|
524 |
||
525 |
/** |
|
0 | 526 |
* @var bool Forces fsockopen() to be used for remote files instead |
527 |
* of cURL, even if a new enough version is installed |
|
528 |
* @see SimplePie::force_fsockopen() |
|
529 |
* @access private |
|
530 |
*/ |
|
531 |
public $force_fsockopen = false; |
|
532 |
||
533 |
/** |
|
534 |
* @var bool Force the given data/URL to be treated as a feed no matter what |
|
535 |
* it appears like |
|
536 |
* @see SimplePie::force_feed() |
|
537 |
* @access private |
|
538 |
*/ |
|
539 |
public $force_feed = false; |
|
540 |
||
541 |
/** |
|
542 |
* @var bool Enable/Disable Caching |
|
543 |
* @see SimplePie::enable_cache() |
|
544 |
* @access private |
|
545 |
*/ |
|
546 |
public $cache = true; |
|
547 |
||
548 |
/** |
|
16 | 549 |
* @var bool Force SimplePie to fallback to expired cache, if enabled, |
550 |
* when feed is unavailable. |
|
551 |
* @see SimplePie::force_cache_fallback() |
|
552 |
* @access private |
|
553 |
*/ |
|
554 |
public $force_cache_fallback = false; |
|
555 |
||
556 |
/** |
|
0 | 557 |
* @var int Cache duration (in seconds) |
558 |
* @see SimplePie::set_cache_duration() |
|
559 |
* @access private |
|
560 |
*/ |
|
561 |
public $cache_duration = 3600; |
|
562 |
||
563 |
/** |
|
564 |
* @var int Auto-discovery cache duration (in seconds) |
|
565 |
* @see SimplePie::set_autodiscovery_cache_duration() |
|
566 |
* @access private |
|
567 |
*/ |
|
568 |
public $autodiscovery_cache_duration = 604800; // 7 Days. |
|
569 |
||
570 |
/** |
|
571 |
* @var string Cache location (relative to executing script) |
|
572 |
* @see SimplePie::set_cache_location() |
|
573 |
* @access private |
|
574 |
*/ |
|
575 |
public $cache_location = './cache'; |
|
576 |
||
577 |
/** |
|
578 |
* @var string Function that creates the cache filename |
|
579 |
* @see SimplePie::set_cache_name_function() |
|
580 |
* @access private |
|
581 |
*/ |
|
582 |
public $cache_name_function = 'md5'; |
|
583 |
||
584 |
/** |
|
585 |
* @var bool Reorder feed by date descending |
|
586 |
* @see SimplePie::enable_order_by_date() |
|
587 |
* @access private |
|
588 |
*/ |
|
589 |
public $order_by_date = true; |
|
590 |
||
591 |
/** |
|
592 |
* @var mixed Force input encoding to be set to the follow value |
|
593 |
* (false, or anything type-cast to false, disables this feature) |
|
594 |
* @see SimplePie::set_input_encoding() |
|
595 |
* @access private |
|
596 |
*/ |
|
597 |
public $input_encoding = false; |
|
598 |
||
599 |
/** |
|
600 |
* @var int Feed Autodiscovery Level |
|
601 |
* @see SimplePie::set_autodiscovery_level() |
|
602 |
* @access private |
|
603 |
*/ |
|
604 |
public $autodiscovery = SIMPLEPIE_LOCATOR_ALL; |
|
605 |
||
606 |
/** |
|
607 |
* Class registry object |
|
608 |
* |
|
609 |
* @var SimplePie_Registry |
|
610 |
*/ |
|
611 |
public $registry; |
|
612 |
||
613 |
/** |
|
614 |
* @var int Maximum number of feeds to check with autodiscovery |
|
615 |
* @see SimplePie::set_max_checked_feeds() |
|
616 |
* @access private |
|
617 |
*/ |
|
618 |
public $max_checked_feeds = 10; |
|
619 |
||
620 |
/** |
|
621 |
* @var array All the feeds found during the autodiscovery process |
|
622 |
* @see SimplePie::get_all_discovered_feeds() |
|
623 |
* @access private |
|
624 |
*/ |
|
625 |
public $all_discovered_feeds = array(); |
|
626 |
||
627 |
/** |
|
628 |
* @var string Web-accessible path to the handler_image.php file. |
|
629 |
* @see SimplePie::set_image_handler() |
|
630 |
* @access private |
|
631 |
*/ |
|
632 |
public $image_handler = ''; |
|
633 |
||
634 |
/** |
|
635 |
* @var array Stores the URLs when multiple feeds are being initialized. |
|
636 |
* @see SimplePie::set_feed_url() |
|
637 |
* @access private |
|
638 |
*/ |
|
639 |
public $multifeed_url = array(); |
|
640 |
||
641 |
/** |
|
642 |
* @var array Stores SimplePie objects when multiple feeds initialized. |
|
643 |
* @access private |
|
644 |
*/ |
|
645 |
public $multifeed_objects = array(); |
|
646 |
||
647 |
/** |
|
648 |
* @var array Stores the get_object_vars() array for use with multifeeds. |
|
649 |
* @see SimplePie::set_feed_url() |
|
650 |
* @access private |
|
651 |
*/ |
|
652 |
public $config_settings = null; |
|
653 |
||
654 |
/** |
|
655 |
* @var integer Stores the number of items to return per-feed with multifeeds. |
|
656 |
* @see SimplePie::set_item_limit() |
|
657 |
* @access private |
|
658 |
*/ |
|
659 |
public $item_limit = 0; |
|
660 |
||
661 |
/** |
|
16 | 662 |
* @var bool Stores if last-modified and/or etag headers were sent with the |
663 |
* request when checking a feed. |
|
664 |
*/ |
|
665 |
public $check_modified = false; |
|
666 |
||
667 |
/** |
|
0 | 668 |
* @var array Stores the default attributes to be stripped by strip_attributes(). |
669 |
* @see SimplePie::strip_attributes() |
|
670 |
* @access private |
|
671 |
*/ |
|
672 |
public $strip_attributes = array('bgsound', 'class', 'expr', 'id', 'style', 'onclick', 'onerror', 'onfinish', 'onmouseover', 'onmouseout', 'onfocus', 'onblur', 'lowsrc', 'dynsrc'); |
|
673 |
||
674 |
/** |
|
16 | 675 |
* @var array Stores the default attributes to add to different tags by add_attributes(). |
676 |
* @see SimplePie::add_attributes() |
|
677 |
* @access private |
|
678 |
*/ |
|
679 |
public $add_attributes = array('audio' => array('preload' => 'none'), 'iframe' => array('sandbox' => 'allow-scripts allow-same-origin'), 'video' => array('preload' => 'none')); |
|
680 |
||
681 |
/** |
|
0 | 682 |
* @var array Stores the default tags to be stripped by strip_htmltags(). |
683 |
* @see SimplePie::strip_htmltags() |
|
684 |
* @access private |
|
685 |
*/ |
|
686 |
public $strip_htmltags = array('base', 'blink', 'body', 'doctype', 'embed', 'font', 'form', 'frame', 'frameset', 'html', 'iframe', 'input', 'marquee', 'meta', 'noscript', 'object', 'param', 'script', 'style'); |
|
687 |
||
688 |
/** |
|
16 | 689 |
* @var bool Should we throw exceptions, or use the old-style error property? |
690 |
* @access private |
|
691 |
*/ |
|
692 |
public $enable_exceptions = false; |
|
693 |
||
694 |
/** |
|
0 | 695 |
* The SimplePie class contains feed level data and options |
696 |
* |
|
697 |
* To use SimplePie, create the SimplePie object with no parameters. You can |
|
698 |
* then set configuration options using the provided methods. After setting |
|
699 |
* them, you must initialise the feed using $feed->init(). At that point the |
|
700 |
* object's methods and properties will be available to you. |
|
701 |
* |
|
702 |
* Previously, it was possible to pass in the feed URL along with cache |
|
703 |
* options directly into the constructor. This has been removed as of 1.3 as |
|
704 |
* it caused a lot of confusion. |
|
705 |
* |
|
706 |
* @since 1.0 Preview Release |
|
707 |
*/ |
|
708 |
public function __construct() |
|
709 |
{ |
|
16 | 710 |
if (version_compare(PHP_VERSION, '5.6', '<')) |
0 | 711 |
{ |
16 | 712 |
trigger_error('Please upgrade to PHP 5.6 or newer.'); |
0 | 713 |
die(); |
714 |
} |
|
715 |
||
716 |
// Other objects, instances created here so we can set options on them |
|
717 |
$this->sanitize = new SimplePie_Sanitize(); |
|
718 |
$this->registry = new SimplePie_Registry(); |
|
719 |
||
720 |
if (func_num_args() > 0) |
|
721 |
{ |
|
722 |
$level = defined('E_USER_DEPRECATED') ? E_USER_DEPRECATED : E_USER_WARNING; |
|
16 | 723 |
trigger_error('Passing parameters to the constructor is no longer supported. Please use set_feed_url(), set_cache_location(), and set_cache_duration() directly.', $level); |
0 | 724 |
|
725 |
$args = func_get_args(); |
|
726 |
switch (count($args)) { |
|
727 |
case 3: |
|
728 |
$this->set_cache_duration($args[2]); |
|
729 |
case 2: |
|
730 |
$this->set_cache_location($args[1]); |
|
731 |
case 1: |
|
732 |
$this->set_feed_url($args[0]); |
|
733 |
$this->init(); |
|
734 |
} |
|
735 |
} |
|
736 |
} |
|
737 |
||
738 |
/** |
|
739 |
* Used for converting object to a string |
|
740 |
*/ |
|
741 |
public function __toString() |
|
742 |
{ |
|
743 |
return md5(serialize($this->data)); |
|
744 |
} |
|
745 |
||
746 |
/** |
|
747 |
* Remove items that link back to this before destroying this object |
|
748 |
*/ |
|
749 |
public function __destruct() |
|
750 |
{ |
|
16 | 751 |
if (!gc_enabled()) |
0 | 752 |
{ |
753 |
if (!empty($this->data['items'])) |
|
754 |
{ |
|
755 |
foreach ($this->data['items'] as $item) |
|
756 |
{ |
|
757 |
$item->__destruct(); |
|
758 |
} |
|
759 |
unset($item, $this->data['items']); |
|
760 |
} |
|
761 |
if (!empty($this->data['ordered_items'])) |
|
762 |
{ |
|
763 |
foreach ($this->data['ordered_items'] as $item) |
|
764 |
{ |
|
765 |
$item->__destruct(); |
|
766 |
} |
|
767 |
unset($item, $this->data['ordered_items']); |
|
768 |
} |
|
769 |
} |
|
770 |
} |
|
771 |
||
772 |
/** |
|
773 |
* Force the given data/URL to be treated as a feed |
|
774 |
* |
|
775 |
* This tells SimplePie to ignore the content-type provided by the server. |
|
776 |
* Be careful when using this option, as it will also disable autodiscovery. |
|
777 |
* |
|
778 |
* @since 1.1 |
|
779 |
* @param bool $enable Force the given data/URL to be treated as a feed |
|
780 |
*/ |
|
781 |
public function force_feed($enable = false) |
|
782 |
{ |
|
783 |
$this->force_feed = (bool) $enable; |
|
784 |
} |
|
785 |
||
786 |
/** |
|
787 |
* Set the URL of the feed you want to parse |
|
788 |
* |
|
789 |
* This allows you to enter the URL of the feed you want to parse, or the |
|
790 |
* website you want to try to use auto-discovery on. This takes priority |
|
791 |
* over any set raw data. |
|
792 |
* |
|
793 |
* You can set multiple feeds to mash together by passing an array instead |
|
794 |
* of a string for the $url. Remember that with each additional feed comes |
|
795 |
* additional processing and resources. |
|
796 |
* |
|
797 |
* @since 1.0 Preview Release |
|
798 |
* @see set_raw_data() |
|
799 |
* @param string|array $url This is the URL (or array of URLs) that you want to parse. |
|
800 |
*/ |
|
801 |
public function set_feed_url($url) |
|
802 |
{ |
|
803 |
$this->multifeed_url = array(); |
|
804 |
if (is_array($url)) |
|
805 |
{ |
|
806 |
foreach ($url as $value) |
|
807 |
{ |
|
808 |
$this->multifeed_url[] = $this->registry->call('Misc', 'fix_protocol', array($value, 1)); |
|
809 |
} |
|
810 |
} |
|
811 |
else |
|
812 |
{ |
|
813 |
$this->feed_url = $this->registry->call('Misc', 'fix_protocol', array($url, 1)); |
|
16 | 814 |
$this->permanent_url = $this->feed_url; |
0 | 815 |
} |
816 |
} |
|
817 |
||
818 |
/** |
|
819 |
* Set an instance of {@see SimplePie_File} to use as a feed |
|
820 |
* |
|
821 |
* @param SimplePie_File &$file |
|
822 |
* @return bool True on success, false on failure |
|
823 |
*/ |
|
824 |
public function set_file(&$file) |
|
825 |
{ |
|
826 |
if ($file instanceof SimplePie_File) |
|
827 |
{ |
|
828 |
$this->feed_url = $file->url; |
|
16 | 829 |
$this->permanent_url = $this->feed_url; |
0 | 830 |
$this->file =& $file; |
831 |
return true; |
|
832 |
} |
|
833 |
return false; |
|
834 |
} |
|
835 |
||
836 |
/** |
|
837 |
* Set the raw XML data to parse |
|
838 |
* |
|
839 |
* Allows you to use a string of RSS/Atom data instead of a remote feed. |
|
840 |
* |
|
841 |
* If you have a feed available as a string in PHP, you can tell SimplePie |
|
842 |
* to parse that data string instead of a remote feed. Any set feed URL |
|
843 |
* takes precedence. |
|
844 |
* |
|
845 |
* @since 1.0 Beta 3 |
|
846 |
* @param string $data RSS or Atom data as a string. |
|
847 |
* @see set_feed_url() |
|
848 |
*/ |
|
849 |
public function set_raw_data($data) |
|
850 |
{ |
|
851 |
$this->raw_data = $data; |
|
852 |
} |
|
853 |
||
854 |
/** |
|
16 | 855 |
* Set the default timeout for fetching remote feeds |
0 | 856 |
* |
857 |
* This allows you to change the maximum time the feed's server to respond |
|
858 |
* and send the feed back. |
|
859 |
* |
|
860 |
* @since 1.0 Beta 3 |
|
861 |
* @param int $timeout The maximum number of seconds to spend waiting to retrieve a feed. |
|
862 |
*/ |
|
863 |
public function set_timeout($timeout = 10) |
|
864 |
{ |
|
865 |
$this->timeout = (int) $timeout; |
|
866 |
} |
|
867 |
||
868 |
/** |
|
16 | 869 |
* Set custom curl options |
870 |
* |
|
871 |
* This allows you to change default curl options |
|
872 |
* |
|
873 |
* @since 1.0 Beta 3 |
|
874 |
* @param array $curl_options Curl options to add to default settings |
|
875 |
*/ |
|
876 |
public function set_curl_options(array $curl_options = array()) |
|
877 |
{ |
|
878 |
$this->curl_options = $curl_options; |
|
879 |
} |
|
880 |
||
881 |
/** |
|
0 | 882 |
* Force SimplePie to use fsockopen() instead of cURL |
883 |
* |
|
884 |
* @since 1.0 Beta 3 |
|
885 |
* @param bool $enable Force fsockopen() to be used |
|
886 |
*/ |
|
887 |
public function force_fsockopen($enable = false) |
|
888 |
{ |
|
889 |
$this->force_fsockopen = (bool) $enable; |
|
890 |
} |
|
891 |
||
892 |
/** |
|
893 |
* Enable/disable caching in SimplePie. |
|
894 |
* |
|
895 |
* This option allows you to disable caching all-together in SimplePie. |
|
896 |
* However, disabling the cache can lead to longer load times. |
|
897 |
* |
|
898 |
* @since 1.0 Preview Release |
|
899 |
* @param bool $enable Enable caching |
|
900 |
*/ |
|
901 |
public function enable_cache($enable = true) |
|
902 |
{ |
|
903 |
$this->cache = (bool) $enable; |
|
904 |
} |
|
905 |
||
906 |
/** |
|
16 | 907 |
* SimplePie to continue to fall back to expired cache, if enabled, when |
908 |
* feed is unavailable. |
|
909 |
* |
|
910 |
* This tells SimplePie to ignore any file errors and fall back to cache |
|
911 |
* instead. This only works if caching is enabled and cached content |
|
912 |
* still exists. |
|
913 |
||
914 |
* @param bool $enable Force use of cache on fail. |
|
915 |
*/ |
|
916 |
public function force_cache_fallback($enable = false) |
|
917 |
{ |
|
918 |
$this->force_cache_fallback= (bool) $enable; |
|
919 |
} |
|
920 |
||
921 |
/** |
|
0 | 922 |
* Set the length of time (in seconds) that the contents of a feed will be |
923 |
* cached |
|
924 |
* |
|
925 |
* @param int $seconds The feed content cache duration |
|
926 |
*/ |
|
927 |
public function set_cache_duration($seconds = 3600) |
|
928 |
{ |
|
929 |
$this->cache_duration = (int) $seconds; |
|
930 |
} |
|
931 |
||
932 |
/** |
|
933 |
* Set the length of time (in seconds) that the autodiscovered feed URL will |
|
934 |
* be cached |
|
935 |
* |
|
936 |
* @param int $seconds The autodiscovered feed URL cache duration. |
|
937 |
*/ |
|
938 |
public function set_autodiscovery_cache_duration($seconds = 604800) |
|
939 |
{ |
|
940 |
$this->autodiscovery_cache_duration = (int) $seconds; |
|
941 |
} |
|
942 |
||
943 |
/** |
|
944 |
* Set the file system location where the cached files should be stored |
|
945 |
* |
|
946 |
* @param string $location The file system location. |
|
947 |
*/ |
|
948 |
public function set_cache_location($location = './cache') |
|
949 |
{ |
|
950 |
$this->cache_location = (string) $location; |
|
951 |
} |
|
952 |
||
953 |
/** |
|
19 | 954 |
* Return the filename (i.e. hash, without path and without extension) of the file to cache a given URL. |
955 |
* @param string $url The URL of the feed to be cached. |
|
956 |
* @return string A filename (i.e. hash, without path and without extension). |
|
957 |
*/ |
|
958 |
public function get_cache_filename($url) |
|
959 |
{ |
|
960 |
// Append custom parameters to the URL to avoid cache pollution in case of multiple calls with different parameters. |
|
961 |
$url .= $this->force_feed ? '#force_feed' : ''; |
|
962 |
$options = array(); |
|
963 |
if ($this->timeout != 10) |
|
964 |
{ |
|
965 |
$options[CURLOPT_TIMEOUT] = $this->timeout; |
|
966 |
} |
|
967 |
if ($this->useragent !== SIMPLEPIE_USERAGENT) |
|
968 |
{ |
|
969 |
$options[CURLOPT_USERAGENT] = $this->useragent; |
|
970 |
} |
|
971 |
if (!empty($this->curl_options)) |
|
972 |
{ |
|
973 |
foreach ($this->curl_options as $k => $v) |
|
974 |
{ |
|
975 |
$options[$k] = $v; |
|
976 |
} |
|
977 |
} |
|
978 |
if (!empty($options)) |
|
979 |
{ |
|
980 |
ksort($options); |
|
981 |
$url .= '#' . urlencode(var_export($options, true)); |
|
982 |
} |
|
983 |
return call_user_func($this->cache_name_function, $url); |
|
984 |
} |
|
985 |
||
986 |
/** |
|
0 | 987 |
* Set whether feed items should be sorted into reverse chronological order |
988 |
* |
|
989 |
* @param bool $enable Sort as reverse chronological order. |
|
990 |
*/ |
|
991 |
public function enable_order_by_date($enable = true) |
|
992 |
{ |
|
993 |
$this->order_by_date = (bool) $enable; |
|
994 |
} |
|
995 |
||
996 |
/** |
|
997 |
* Set the character encoding used to parse the feed |
|
998 |
* |
|
999 |
* This overrides the encoding reported by the feed, however it will fall |
|
1000 |
* back to the normal encoding detection if the override fails |
|
1001 |
* |
|
1002 |
* @param string $encoding Character encoding |
|
1003 |
*/ |
|
1004 |
public function set_input_encoding($encoding = false) |
|
1005 |
{ |
|
1006 |
if ($encoding) |
|
1007 |
{ |
|
1008 |
$this->input_encoding = (string) $encoding; |
|
1009 |
} |
|
1010 |
else |
|
1011 |
{ |
|
1012 |
$this->input_encoding = false; |
|
1013 |
} |
|
1014 |
} |
|
1015 |
||
1016 |
/** |
|
1017 |
* Set how much feed autodiscovery to do |
|
1018 |
* |
|
1019 |
* @see SIMPLEPIE_LOCATOR_NONE |
|
1020 |
* @see SIMPLEPIE_LOCATOR_AUTODISCOVERY |
|
1021 |
* @see SIMPLEPIE_LOCATOR_LOCAL_EXTENSION |
|
1022 |
* @see SIMPLEPIE_LOCATOR_LOCAL_BODY |
|
1023 |
* @see SIMPLEPIE_LOCATOR_REMOTE_EXTENSION |
|
1024 |
* @see SIMPLEPIE_LOCATOR_REMOTE_BODY |
|
1025 |
* @see SIMPLEPIE_LOCATOR_ALL |
|
1026 |
* @param int $level Feed Autodiscovery Level (level can be a combination of the above constants, see bitwise OR operator) |
|
1027 |
*/ |
|
1028 |
public function set_autodiscovery_level($level = SIMPLEPIE_LOCATOR_ALL) |
|
1029 |
{ |
|
1030 |
$this->autodiscovery = (int) $level; |
|
1031 |
} |
|
1032 |
||
1033 |
/** |
|
1034 |
* Get the class registry |
|
1035 |
* |
|
1036 |
* Use this to override SimplePie's default classes |
|
1037 |
* @see SimplePie_Registry |
|
1038 |
* @return SimplePie_Registry |
|
1039 |
*/ |
|
1040 |
public function &get_registry() |
|
1041 |
{ |
|
1042 |
return $this->registry; |
|
1043 |
} |
|
1044 |
||
1045 |
/**#@+ |
|
1046 |
* Useful when you are overloading or extending SimplePie's default classes. |
|
1047 |
* |
|
1048 |
* @deprecated Use {@see get_registry()} instead |
|
1049 |
* @link http://php.net/manual/en/language.oop5.basic.php#language.oop5.basic.extends PHP5 extends documentation |
|
1050 |
* @param string $class Name of custom class |
|
1051 |
* @return boolean True on success, false otherwise |
|
1052 |
*/ |
|
1053 |
/** |
|
1054 |
* Set which class SimplePie uses for caching |
|
1055 |
*/ |
|
1056 |
public function set_cache_class($class = 'SimplePie_Cache') |
|
1057 |
{ |
|
1058 |
return $this->registry->register('Cache', $class, true); |
|
1059 |
} |
|
1060 |
||
1061 |
/** |
|
1062 |
* Set which class SimplePie uses for auto-discovery |
|
1063 |
*/ |
|
1064 |
public function set_locator_class($class = 'SimplePie_Locator') |
|
1065 |
{ |
|
1066 |
return $this->registry->register('Locator', $class, true); |
|
1067 |
} |
|
1068 |
||
1069 |
/** |
|
1070 |
* Set which class SimplePie uses for XML parsing |
|
1071 |
*/ |
|
1072 |
public function set_parser_class($class = 'SimplePie_Parser') |
|
1073 |
{ |
|
1074 |
return $this->registry->register('Parser', $class, true); |
|
1075 |
} |
|
1076 |
||
1077 |
/** |
|
1078 |
* Set which class SimplePie uses for remote file fetching |
|
1079 |
*/ |
|
1080 |
public function set_file_class($class = 'SimplePie_File') |
|
1081 |
{ |
|
1082 |
return $this->registry->register('File', $class, true); |
|
1083 |
} |
|
1084 |
||
1085 |
/** |
|
1086 |
* Set which class SimplePie uses for data sanitization |
|
1087 |
*/ |
|
1088 |
public function set_sanitize_class($class = 'SimplePie_Sanitize') |
|
1089 |
{ |
|
1090 |
return $this->registry->register('Sanitize', $class, true); |
|
1091 |
} |
|
1092 |
||
1093 |
/** |
|
1094 |
* Set which class SimplePie uses for handling feed items |
|
1095 |
*/ |
|
1096 |
public function set_item_class($class = 'SimplePie_Item') |
|
1097 |
{ |
|
1098 |
return $this->registry->register('Item', $class, true); |
|
1099 |
} |
|
1100 |
||
1101 |
/** |
|
1102 |
* Set which class SimplePie uses for handling author data |
|
1103 |
*/ |
|
1104 |
public function set_author_class($class = 'SimplePie_Author') |
|
1105 |
{ |
|
1106 |
return $this->registry->register('Author', $class, true); |
|
1107 |
} |
|
1108 |
||
1109 |
/** |
|
1110 |
* Set which class SimplePie uses for handling category data |
|
1111 |
*/ |
|
1112 |
public function set_category_class($class = 'SimplePie_Category') |
|
1113 |
{ |
|
1114 |
return $this->registry->register('Category', $class, true); |
|
1115 |
} |
|
1116 |
||
1117 |
/** |
|
1118 |
* Set which class SimplePie uses for feed enclosures |
|
1119 |
*/ |
|
1120 |
public function set_enclosure_class($class = 'SimplePie_Enclosure') |
|
1121 |
{ |
|
1122 |
return $this->registry->register('Enclosure', $class, true); |
|
1123 |
} |
|
1124 |
||
1125 |
/** |
|
1126 |
* Set which class SimplePie uses for `<media:text>` captions |
|
1127 |
*/ |
|
1128 |
public function set_caption_class($class = 'SimplePie_Caption') |
|
1129 |
{ |
|
1130 |
return $this->registry->register('Caption', $class, true); |
|
1131 |
} |
|
1132 |
||
1133 |
/** |
|
1134 |
* Set which class SimplePie uses for `<media:copyright>` |
|
1135 |
*/ |
|
1136 |
public function set_copyright_class($class = 'SimplePie_Copyright') |
|
1137 |
{ |
|
1138 |
return $this->registry->register('Copyright', $class, true); |
|
1139 |
} |
|
1140 |
||
1141 |
/** |
|
1142 |
* Set which class SimplePie uses for `<media:credit>` |
|
1143 |
*/ |
|
1144 |
public function set_credit_class($class = 'SimplePie_Credit') |
|
1145 |
{ |
|
1146 |
return $this->registry->register('Credit', $class, true); |
|
1147 |
} |
|
1148 |
||
1149 |
/** |
|
1150 |
* Set which class SimplePie uses for `<media:rating>` |
|
1151 |
*/ |
|
1152 |
public function set_rating_class($class = 'SimplePie_Rating') |
|
1153 |
{ |
|
1154 |
return $this->registry->register('Rating', $class, true); |
|
1155 |
} |
|
1156 |
||
1157 |
/** |
|
1158 |
* Set which class SimplePie uses for `<media:restriction>` |
|
1159 |
*/ |
|
1160 |
public function set_restriction_class($class = 'SimplePie_Restriction') |
|
1161 |
{ |
|
1162 |
return $this->registry->register('Restriction', $class, true); |
|
1163 |
} |
|
1164 |
||
1165 |
/** |
|
1166 |
* Set which class SimplePie uses for content-type sniffing |
|
1167 |
*/ |
|
1168 |
public function set_content_type_sniffer_class($class = 'SimplePie_Content_Type_Sniffer') |
|
1169 |
{ |
|
1170 |
return $this->registry->register('Content_Type_Sniffer', $class, true); |
|
1171 |
} |
|
1172 |
||
1173 |
/** |
|
1174 |
* Set which class SimplePie uses item sources |
|
1175 |
*/ |
|
1176 |
public function set_source_class($class = 'SimplePie_Source') |
|
1177 |
{ |
|
1178 |
return $this->registry->register('Source', $class, true); |
|
1179 |
} |
|
1180 |
/**#@-*/ |
|
1181 |
||
1182 |
/** |
|
1183 |
* Set the user agent string |
|
1184 |
* |
|
1185 |
* @param string $ua New user agent string. |
|
1186 |
*/ |
|
1187 |
public function set_useragent($ua = SIMPLEPIE_USERAGENT) |
|
1188 |
{ |
|
1189 |
$this->useragent = (string) $ua; |
|
1190 |
} |
|
1191 |
||
1192 |
/** |
|
1193 |
* Set callback function to create cache filename with |
|
1194 |
* |
|
1195 |
* @param mixed $function Callback function |
|
1196 |
*/ |
|
1197 |
public function set_cache_name_function($function = 'md5') |
|
1198 |
{ |
|
1199 |
if (is_callable($function)) |
|
1200 |
{ |
|
1201 |
$this->cache_name_function = $function; |
|
1202 |
} |
|
1203 |
} |
|
1204 |
||
1205 |
/** |
|
1206 |
* Set options to make SP as fast as possible |
|
1207 |
* |
|
1208 |
* Forgoes a substantial amount of data sanitization in favor of speed. This |
|
1209 |
* turns SimplePie into a dumb parser of feeds. |
|
1210 |
* |
|
1211 |
* @param bool $set Whether to set them or not |
|
1212 |
*/ |
|
1213 |
public function set_stupidly_fast($set = false) |
|
1214 |
{ |
|
1215 |
if ($set) |
|
1216 |
{ |
|
1217 |
$this->enable_order_by_date(false); |
|
1218 |
$this->remove_div(false); |
|
1219 |
$this->strip_comments(false); |
|
1220 |
$this->strip_htmltags(false); |
|
1221 |
$this->strip_attributes(false); |
|
16 | 1222 |
$this->add_attributes(false); |
0 | 1223 |
$this->set_image_handler(false); |
19 | 1224 |
$this->set_https_domains(array()); |
0 | 1225 |
} |
1226 |
} |
|
1227 |
||
1228 |
/** |
|
1229 |
* Set maximum number of feeds to check with autodiscovery |
|
1230 |
* |
|
1231 |
* @param int $max Maximum number of feeds to check |
|
1232 |
*/ |
|
1233 |
public function set_max_checked_feeds($max = 10) |
|
1234 |
{ |
|
1235 |
$this->max_checked_feeds = (int) $max; |
|
1236 |
} |
|
1237 |
||
1238 |
public function remove_div($enable = true) |
|
1239 |
{ |
|
1240 |
$this->sanitize->remove_div($enable); |
|
1241 |
} |
|
1242 |
||
1243 |
public function strip_htmltags($tags = '', $encode = null) |
|
1244 |
{ |
|
1245 |
if ($tags === '') |
|
1246 |
{ |
|
1247 |
$tags = $this->strip_htmltags; |
|
1248 |
} |
|
1249 |
$this->sanitize->strip_htmltags($tags); |
|
1250 |
if ($encode !== null) |
|
1251 |
{ |
|
1252 |
$this->sanitize->encode_instead_of_strip($tags); |
|
1253 |
} |
|
1254 |
} |
|
1255 |
||
1256 |
public function encode_instead_of_strip($enable = true) |
|
1257 |
{ |
|
1258 |
$this->sanitize->encode_instead_of_strip($enable); |
|
1259 |
} |
|
1260 |
||
1261 |
public function strip_attributes($attribs = '') |
|
1262 |
{ |
|
1263 |
if ($attribs === '') |
|
1264 |
{ |
|
1265 |
$attribs = $this->strip_attributes; |
|
1266 |
} |
|
1267 |
$this->sanitize->strip_attributes($attribs); |
|
1268 |
} |
|
1269 |
||
16 | 1270 |
public function add_attributes($attribs = '') |
1271 |
{ |
|
1272 |
if ($attribs === '') |
|
1273 |
{ |
|
1274 |
$attribs = $this->add_attributes; |
|
1275 |
} |
|
1276 |
$this->sanitize->add_attributes($attribs); |
|
1277 |
} |
|
1278 |
||
0 | 1279 |
/** |
1280 |
* Set the output encoding |
|
1281 |
* |
|
1282 |
* Allows you to override SimplePie's output to match that of your webpage. |
|
1283 |
* This is useful for times when your webpages are not being served as |
|
16 | 1284 |
* UTF-8. This setting will be obeyed by {@see handle_content_type()}, and |
0 | 1285 |
* is similar to {@see set_input_encoding()}. |
1286 |
* |
|
1287 |
* It should be noted, however, that not all character encodings can support |
|
16 | 1288 |
* all characters. If your page is being served as ISO-8859-1 and you try |
0 | 1289 |
* to display a Japanese feed, you'll likely see garbled characters. |
1290 |
* Because of this, it is highly recommended to ensure that your webpages |
|
1291 |
* are served as UTF-8. |
|
1292 |
* |
|
1293 |
* The number of supported character encodings depends on whether your web |
|
1294 |
* host supports {@link http://php.net/mbstring mbstring}, |
|
1295 |
* {@link http://php.net/iconv iconv}, or both. See |
|
1296 |
* {@link http://simplepie.org/wiki/faq/Supported_Character_Encodings} for |
|
1297 |
* more information. |
|
1298 |
* |
|
1299 |
* @param string $encoding |
|
1300 |
*/ |
|
1301 |
public function set_output_encoding($encoding = 'UTF-8') |
|
1302 |
{ |
|
1303 |
$this->sanitize->set_output_encoding($encoding); |
|
1304 |
} |
|
1305 |
||
1306 |
public function strip_comments($strip = false) |
|
1307 |
{ |
|
1308 |
$this->sanitize->strip_comments($strip); |
|
1309 |
} |
|
1310 |
||
1311 |
/** |
|
1312 |
* Set element/attribute key/value pairs of HTML attributes |
|
1313 |
* containing URLs that need to be resolved relative to the feed |
|
1314 |
* |
|
1315 |
* Defaults to |a|@href, |area|@href, |blockquote|@cite, |del|@cite, |
|
1316 |
* |form|@action, |img|@longdesc, |img|@src, |input|@src, |ins|@cite, |
|
1317 |
* |q|@cite |
|
1318 |
* |
|
1319 |
* @since 1.0 |
|
1320 |
* @param array|null $element_attribute Element/attribute key/value pairs, null for default |
|
1321 |
*/ |
|
1322 |
public function set_url_replacements($element_attribute = null) |
|
1323 |
{ |
|
1324 |
$this->sanitize->set_url_replacements($element_attribute); |
|
1325 |
} |
|
1326 |
||
1327 |
/** |
|
19 | 1328 |
* Set the list of domains for which to force HTTPS. |
1329 |
* @see SimplePie_Sanitize::set_https_domains() |
|
1330 |
* @param array List of HTTPS domains. Example array('biz', 'example.com', 'example.org', 'www.example.net'). |
|
1331 |
*/ |
|
1332 |
public function set_https_domains($domains = array()) |
|
1333 |
{ |
|
1334 |
if (is_array($domains)) |
|
1335 |
{ |
|
1336 |
$this->sanitize->set_https_domains($domains); |
|
1337 |
} |
|
1338 |
} |
|
1339 |
||
1340 |
/** |
|
0 | 1341 |
* Set the handler to enable the display of cached images. |
1342 |
* |
|
16 | 1343 |
* @param string $page Web-accessible path to the handler_image.php file. |
1344 |
* @param string $qs The query string that the value should be passed to. |
|
0 | 1345 |
*/ |
1346 |
public function set_image_handler($page = false, $qs = 'i') |
|
1347 |
{ |
|
1348 |
if ($page !== false) |
|
1349 |
{ |
|
1350 |
$this->sanitize->set_image_handler($page . '?' . $qs . '='); |
|
1351 |
} |
|
1352 |
else |
|
1353 |
{ |
|
1354 |
$this->image_handler = ''; |
|
1355 |
} |
|
1356 |
} |
|
1357 |
||
1358 |
/** |
|
1359 |
* Set the limit for items returned per-feed with multifeeds |
|
1360 |
* |
|
1361 |
* @param integer $limit The maximum number of items to return. |
|
1362 |
*/ |
|
1363 |
public function set_item_limit($limit = 0) |
|
1364 |
{ |
|
1365 |
$this->item_limit = (int) $limit; |
|
1366 |
} |
|
1367 |
||
1368 |
/** |
|
16 | 1369 |
* Enable throwing exceptions |
1370 |
* |
|
1371 |
* @param boolean $enable Should we throw exceptions, or use the old-style error property? |
|
1372 |
*/ |
|
1373 |
public function enable_exceptions($enable = true) |
|
1374 |
{ |
|
1375 |
$this->enable_exceptions = $enable; |
|
1376 |
} |
|
1377 |
||
1378 |
/** |
|
0 | 1379 |
* Initialize the feed object |
1380 |
* |
|
16 | 1381 |
* This is what makes everything happen. Period. This is where all of the |
0 | 1382 |
* configuration options get processed, feeds are fetched, cached, and |
1383 |
* parsed, and all of that other good stuff. |
|
1384 |
* |
|
1385 |
* @return boolean True if successful, false otherwise |
|
1386 |
*/ |
|
1387 |
public function init() |
|
1388 |
{ |
|
1389 |
// Check absolute bare minimum requirements. |
|
1390 |
if (!extension_loaded('xml') || !extension_loaded('pcre')) |
|
1391 |
{ |
|
16 | 1392 |
$this->error = 'XML or PCRE extensions not loaded!'; |
0 | 1393 |
return false; |
1394 |
} |
|
1395 |
// Then check the xml extension is sane (i.e., libxml 2.7.x issue on PHP < 5.2.9 and libxml 2.7.0 to 2.7.2 on any version) if we don't have xmlreader. |
|
1396 |
elseif (!extension_loaded('xmlreader')) |
|
1397 |
{ |
|
1398 |
static $xml_is_sane = null; |
|
1399 |
if ($xml_is_sane === null) |
|
1400 |
{ |
|
1401 |
$parser_check = xml_parser_create(); |
|
1402 |
xml_parse_into_struct($parser_check, '<foo>&</foo>', $values); |
|
1403 |
xml_parser_free($parser_check); |
|
1404 |
$xml_is_sane = isset($values[0]['value']); |
|
1405 |
} |
|
1406 |
if (!$xml_is_sane) |
|
1407 |
{ |
|
1408 |
return false; |
|
1409 |
} |
|
1410 |
} |
|
1411 |
||
16 | 1412 |
// The default sanitize class gets set in the constructor, check if it has |
1413 |
// changed. |
|
1414 |
if ($this->registry->get_class('Sanitize') !== 'SimplePie_Sanitize') { |
|
1415 |
$this->sanitize = $this->registry->create('Sanitize'); |
|
1416 |
} |
|
0 | 1417 |
if (method_exists($this->sanitize, 'set_registry')) |
1418 |
{ |
|
1419 |
$this->sanitize->set_registry($this->registry); |
|
1420 |
} |
|
1421 |
||
1422 |
// Pass whatever was set with config options over to the sanitizer. |
|
1423 |
// Pass the classes in for legacy support; new classes should use the registry instead |
|
1424 |
$this->sanitize->pass_cache_data($this->cache, $this->cache_location, $this->cache_name_function, $this->registry->get_class('Cache')); |
|
16 | 1425 |
$this->sanitize->pass_file_data($this->registry->get_class('File'), $this->timeout, $this->useragent, $this->force_fsockopen, $this->curl_options); |
0 | 1426 |
|
1427 |
if (!empty($this->multifeed_url)) |
|
1428 |
{ |
|
1429 |
$i = 0; |
|
1430 |
$success = 0; |
|
1431 |
$this->multifeed_objects = array(); |
|
1432 |
$this->error = array(); |
|
1433 |
foreach ($this->multifeed_url as $url) |
|
1434 |
{ |
|
1435 |
$this->multifeed_objects[$i] = clone $this; |
|
1436 |
$this->multifeed_objects[$i]->set_feed_url($url); |
|
1437 |
$single_success = $this->multifeed_objects[$i]->init(); |
|
1438 |
$success |= $single_success; |
|
1439 |
if (!$single_success) |
|
1440 |
{ |
|
1441 |
$this->error[$i] = $this->multifeed_objects[$i]->error(); |
|
1442 |
} |
|
1443 |
$i++; |
|
1444 |
} |
|
1445 |
return (bool) $success; |
|
1446 |
} |
|
1447 |
elseif ($this->feed_url === null && $this->raw_data === null) |
|
1448 |
{ |
|
1449 |
return false; |
|
1450 |
} |
|
1451 |
||
1452 |
$this->error = null; |
|
1453 |
$this->data = array(); |
|
16 | 1454 |
$this->check_modified = false; |
0 | 1455 |
$this->multifeed_objects = array(); |
1456 |
$cache = false; |
|
1457 |
||
1458 |
if ($this->feed_url !== null) |
|
1459 |
{ |
|
1460 |
$parsed_feed_url = $this->registry->call('Misc', 'parse_url', array($this->feed_url)); |
|
1461 |
||
1462 |
// Decide whether to enable caching |
|
1463 |
if ($this->cache && $parsed_feed_url['scheme'] !== '') |
|
1464 |
{ |
|
19 | 1465 |
$filename = $this->get_cache_filename($this->feed_url); |
1466 |
$cache = $this->registry->call('Cache', 'get_handler', array($this->cache_location, $filename, 'spc')); |
|
0 | 1467 |
} |
1468 |
||
1469 |
// Fetch the data via SimplePie_File into $this->raw_data |
|
1470 |
if (($fetched = $this->fetch_data($cache)) === true) |
|
1471 |
{ |
|
1472 |
return true; |
|
1473 |
} |
|
1474 |
elseif ($fetched === false) { |
|
1475 |
return false; |
|
1476 |
} |
|
1477 |
||
1478 |
list($headers, $sniffed) = $fetched; |
|
1479 |
} |
|
1480 |
||
16 | 1481 |
// Empty response check |
1482 |
if(empty($this->raw_data)){ |
|
1483 |
$this->error = "A feed could not be found at `$this->feed_url`. Empty body."; |
|
1484 |
$this->registry->call('Misc', 'error', array($this->error, E_USER_NOTICE, __FILE__, __LINE__)); |
|
1485 |
return false; |
|
1486 |
} |
|
1487 |
||
0 | 1488 |
// Set up array of possible encodings |
1489 |
$encodings = array(); |
|
1490 |
||
1491 |
// First check to see if input has been overridden. |
|
1492 |
if ($this->input_encoding !== false) |
|
1493 |
{ |
|
16 | 1494 |
$encodings[] = strtoupper($this->input_encoding); |
0 | 1495 |
} |
1496 |
||
1497 |
$application_types = array('application/xml', 'application/xml-dtd', 'application/xml-external-parsed-entity'); |
|
1498 |
$text_types = array('text/xml', 'text/xml-external-parsed-entity'); |
|
1499 |
||
1500 |
// RFC 3023 (only applies to sniffed content) |
|
1501 |
if (isset($sniffed)) |
|
1502 |
{ |
|
1503 |
if (in_array($sniffed, $application_types) || substr($sniffed, 0, 12) === 'application/' && substr($sniffed, -4) === '+xml') |
|
1504 |
{ |
|
1505 |
if (isset($headers['content-type']) && preg_match('/;\x20?charset=([^;]*)/i', $headers['content-type'], $charset)) |
|
1506 |
{ |
|
1507 |
$encodings[] = strtoupper($charset[1]); |
|
1508 |
} |
|
1509 |
$encodings = array_merge($encodings, $this->registry->call('Misc', 'xml_encoding', array($this->raw_data, &$this->registry))); |
|
1510 |
$encodings[] = 'UTF-8'; |
|
1511 |
} |
|
1512 |
elseif (in_array($sniffed, $text_types) || substr($sniffed, 0, 5) === 'text/' && substr($sniffed, -4) === '+xml') |
|
1513 |
{ |
|
1514 |
if (isset($headers['content-type']) && preg_match('/;\x20?charset=([^;]*)/i', $headers['content-type'], $charset)) |
|
1515 |
{ |
|
16 | 1516 |
$encodings[] = strtoupper($charset[1]); |
0 | 1517 |
} |
1518 |
$encodings[] = 'US-ASCII'; |
|
1519 |
} |
|
1520 |
// Text MIME-type default |
|
1521 |
elseif (substr($sniffed, 0, 5) === 'text/') |
|
1522 |
{ |
|
16 | 1523 |
$encodings[] = 'UTF-8'; |
0 | 1524 |
} |
1525 |
} |
|
1526 |
||
1527 |
// Fallback to XML 1.0 Appendix F.1/UTF-8/ISO-8859-1 |
|
1528 |
$encodings = array_merge($encodings, $this->registry->call('Misc', 'xml_encoding', array($this->raw_data, &$this->registry))); |
|
1529 |
$encodings[] = 'UTF-8'; |
|
1530 |
$encodings[] = 'ISO-8859-1'; |
|
1531 |
||
1532 |
// There's no point in trying an encoding twice |
|
1533 |
$encodings = array_unique($encodings); |
|
1534 |
||
1535 |
// Loop through each possible encoding, till we return something, or run out of possibilities |
|
1536 |
foreach ($encodings as $encoding) |
|
1537 |
{ |
|
1538 |
// Change the encoding to UTF-8 (as we always use UTF-8 internally) |
|
1539 |
if ($utf8_data = $this->registry->call('Misc', 'change_encoding', array($this->raw_data, $encoding, 'UTF-8'))) |
|
1540 |
{ |
|
1541 |
// Create new parser |
|
1542 |
$parser = $this->registry->create('Parser'); |
|
1543 |
||
1544 |
// If it's parsed fine |
|
16 | 1545 |
if ($parser->parse($utf8_data, 'UTF-8', $this->permanent_url)) |
0 | 1546 |
{ |
1547 |
$this->data = $parser->get_data(); |
|
1548 |
if (!($this->get_type() & ~SIMPLEPIE_TYPE_NONE)) |
|
1549 |
{ |
|
16 | 1550 |
$this->error = "A feed could not be found at `$this->feed_url`. This does not appear to be a valid RSS or Atom feed."; |
0 | 1551 |
$this->registry->call('Misc', 'error', array($this->error, E_USER_NOTICE, __FILE__, __LINE__)); |
1552 |
return false; |
|
1553 |
} |
|
1554 |
||
1555 |
if (isset($headers)) |
|
1556 |
{ |
|
1557 |
$this->data['headers'] = $headers; |
|
1558 |
} |
|
1559 |
$this->data['build'] = SIMPLEPIE_BUILD; |
|
1560 |
||
1561 |
// Cache the file if caching is enabled |
|
1562 |
if ($cache && !$cache->save($this)) |
|
1563 |
{ |
|
16 | 1564 |
trigger_error("$this->cache_location is not writable. Make sure you've set the correct relative or absolute path, and that the location is server-writable.", E_USER_WARNING); |
0 | 1565 |
} |
1566 |
return true; |
|
1567 |
} |
|
1568 |
} |
|
1569 |
} |
|
1570 |
||
1571 |
if (isset($parser)) |
|
1572 |
{ |
|
1573 |
// We have an error, just set SimplePie_Misc::error to it and quit |
|
16 | 1574 |
$this->error = $this->feed_url; |
1575 |
$this->error .= sprintf(' is invalid XML, likely due to invalid characters. XML error: %s at line %d, column %d', $parser->get_error_string(), $parser->get_current_line(), $parser->get_current_column()); |
|
0 | 1576 |
} |
1577 |
else |
|
1578 |
{ |
|
16 | 1579 |
$this->error = 'The data could not be converted to UTF-8.'; |
1580 |
if (!extension_loaded('mbstring') && !extension_loaded('iconv') && !class_exists('\UConverter')) { |
|
1581 |
$this->error .= ' You MUST have either the iconv, mbstring or intl (PHP 5.5+) extension installed and enabled.'; |
|
1582 |
} else { |
|
1583 |
$missingExtensions = array(); |
|
1584 |
if (!extension_loaded('iconv')) { |
|
1585 |
$missingExtensions[] = 'iconv'; |
|
1586 |
} |
|
1587 |
if (!extension_loaded('mbstring')) { |
|
1588 |
$missingExtensions[] = 'mbstring'; |
|
1589 |
} |
|
1590 |
if (!class_exists('\UConverter')) { |
|
1591 |
$missingExtensions[] = 'intl (PHP 5.5+)'; |
|
1592 |
} |
|
1593 |
$this->error .= ' Try installing/enabling the ' . implode(' or ', $missingExtensions) . ' extension.'; |
|
1594 |
} |
|
0 | 1595 |
} |
1596 |
||
1597 |
$this->registry->call('Misc', 'error', array($this->error, E_USER_NOTICE, __FILE__, __LINE__)); |
|
1598 |
||
1599 |
return false; |
|
1600 |
} |
|
1601 |
||
1602 |
/** |
|
1603 |
* Fetch the data via SimplePie_File |
|
1604 |
* |
|
1605 |
* If the data is already cached, attempt to fetch it from there instead |
|
19 | 1606 |
* @param SimplePie_Cache_Base|false $cache Cache handler, or false to not load from the cache |
0 | 1607 |
* @return array|true Returns true if the data was loaded from the cache, or an array of HTTP headers and sniffed type |
1608 |
*/ |
|
1609 |
protected function fetch_data(&$cache) |
|
1610 |
{ |
|
1611 |
// If it's enabled, use the cache |
|
1612 |
if ($cache) |
|
1613 |
{ |
|
1614 |
// Load the Cache |
|
1615 |
$this->data = $cache->load(); |
|
1616 |
if (!empty($this->data)) |
|
1617 |
{ |
|
1618 |
// If the cache is for an outdated build of SimplePie |
|
1619 |
if (!isset($this->data['build']) || $this->data['build'] !== SIMPLEPIE_BUILD) |
|
1620 |
{ |
|
1621 |
$cache->unlink(); |
|
1622 |
$this->data = array(); |
|
1623 |
} |
|
1624 |
// If we've hit a collision just rerun it with caching disabled |
|
1625 |
elseif (isset($this->data['url']) && $this->data['url'] !== $this->feed_url) |
|
1626 |
{ |
|
1627 |
$cache = false; |
|
1628 |
$this->data = array(); |
|
1629 |
} |
|
1630 |
// If we've got a non feed_url stored (if the page isn't actually a feed, or is a redirect) use that URL. |
|
1631 |
elseif (isset($this->data['feed_url'])) |
|
1632 |
{ |
|
1633 |
// If the autodiscovery cache is still valid use it. |
|
1634 |
if ($cache->mtime() + $this->autodiscovery_cache_duration > time()) |
|
1635 |
{ |
|
1636 |
// Do not need to do feed autodiscovery yet. |
|
1637 |
if ($this->data['feed_url'] !== $this->data['url']) |
|
1638 |
{ |
|
1639 |
$this->set_feed_url($this->data['feed_url']); |
|
1640 |
return $this->init(); |
|
1641 |
} |
|
1642 |
||
1643 |
$cache->unlink(); |
|
1644 |
$this->data = array(); |
|
1645 |
} |
|
1646 |
} |
|
1647 |
// Check if the cache has been updated |
|
1648 |
elseif ($cache->mtime() + $this->cache_duration < time()) |
|
1649 |
{ |
|
16 | 1650 |
// Want to know if we tried to send last-modified and/or etag headers |
1651 |
// when requesting this file. (Note that it's up to the file to |
|
1652 |
// support this, but we don't always send the headers either.) |
|
1653 |
$this->check_modified = true; |
|
0 | 1654 |
if (isset($this->data['headers']['last-modified']) || isset($this->data['headers']['etag'])) |
1655 |
{ |
|
1656 |
$headers = array( |
|
1657 |
'Accept' => 'application/atom+xml, application/rss+xml, application/rdf+xml;q=0.9, application/xml;q=0.8, text/xml;q=0.8, text/html;q=0.7, unknown/unknown;q=0.1, application/unknown;q=0.1, */*;q=0.1', |
|
1658 |
); |
|
1659 |
if (isset($this->data['headers']['last-modified'])) |
|
1660 |
{ |
|
1661 |
$headers['if-modified-since'] = $this->data['headers']['last-modified']; |
|
1662 |
} |
|
1663 |
if (isset($this->data['headers']['etag'])) |
|
1664 |
{ |
|
1665 |
$headers['if-none-match'] = $this->data['headers']['etag']; |
|
1666 |
} |
|
1667 |
||
16 | 1668 |
$file = $this->registry->create('File', array($this->feed_url, $this->timeout/10, 5, $headers, $this->useragent, $this->force_fsockopen, $this->curl_options)); |
19 | 1669 |
$this->status_code = $file->status_code; |
0 | 1670 |
|
1671 |
if ($file->success) |
|
1672 |
{ |
|
1673 |
if ($file->status_code === 304) |
|
1674 |
{ |
|
16 | 1675 |
// Set raw_data to false here too, to signify that the cache |
1676 |
// is still valid. |
|
1677 |
$this->raw_data = false; |
|
0 | 1678 |
$cache->touch(); |
1679 |
return true; |
|
1680 |
} |
|
1681 |
} |
|
1682 |
else |
|
1683 |
{ |
|
16 | 1684 |
$this->check_modified = false; |
1685 |
if($this->force_cache_fallback) |
|
1686 |
{ |
|
1687 |
$cache->touch(); |
|
1688 |
return true; |
|
1689 |
} |
|
1690 |
||
0 | 1691 |
unset($file); |
1692 |
} |
|
1693 |
} |
|
1694 |
} |
|
1695 |
// If the cache is still valid, just return true |
|
1696 |
else |
|
1697 |
{ |
|
1698 |
$this->raw_data = false; |
|
1699 |
return true; |
|
1700 |
} |
|
1701 |
} |
|
1702 |
// If the cache is empty, delete it |
|
1703 |
else |
|
1704 |
{ |
|
1705 |
$cache->unlink(); |
|
1706 |
$this->data = array(); |
|
1707 |
} |
|
1708 |
} |
|
1709 |
// If we don't already have the file (it'll only exist if we've opened it to check if the cache has been modified), open it. |
|
1710 |
if (!isset($file)) |
|
1711 |
{ |
|
1712 |
if ($this->file instanceof SimplePie_File && $this->file->url === $this->feed_url) |
|
1713 |
{ |
|
1714 |
$file =& $this->file; |
|
1715 |
} |
|
1716 |
else |
|
1717 |
{ |
|
1718 |
$headers = array( |
|
1719 |
'Accept' => 'application/atom+xml, application/rss+xml, application/rdf+xml;q=0.9, application/xml;q=0.8, text/xml;q=0.8, text/html;q=0.7, unknown/unknown;q=0.1, application/unknown;q=0.1, */*;q=0.1', |
|
1720 |
); |
|
16 | 1721 |
$file = $this->registry->create('File', array($this->feed_url, $this->timeout, 5, $headers, $this->useragent, $this->force_fsockopen, $this->curl_options)); |
0 | 1722 |
} |
1723 |
} |
|
19 | 1724 |
$this->status_code = $file->status_code; |
1725 |
||
0 | 1726 |
// If the file connection has an error, set SimplePie::error to that and quit |
1727 |
if (!$file->success && !($file->method & SIMPLEPIE_FILE_SOURCE_REMOTE === 0 || ($file->status_code === 200 || $file->status_code > 206 && $file->status_code < 300))) |
|
1728 |
{ |
|
1729 |
$this->error = $file->error; |
|
1730 |
return !empty($this->data); |
|
1731 |
} |
|
1732 |
||
1733 |
if (!$this->force_feed) |
|
1734 |
{ |
|
1735 |
// Check if the supplied URL is a feed, if it isn't, look for it. |
|
16 | 1736 |
$locate = $this->registry->create('Locator', array(&$file, $this->timeout, $this->useragent, $this->max_checked_feeds, $this->force_fsockopen, $this->curl_options)); |
0 | 1737 |
|
1738 |
if (!$locate->is_feed($file)) |
|
1739 |
{ |
|
16 | 1740 |
$copyStatusCode = $file->status_code; |
1741 |
$copyContentType = $file->headers['content-type']; |
|
0 | 1742 |
try |
1743 |
{ |
|
16 | 1744 |
$microformats = false; |
1745 |
if (class_exists('DOMXpath') && function_exists('Mf2\parse')) { |
|
1746 |
$doc = new DOMDocument(); |
|
1747 |
@$doc->loadHTML($file->body); |
|
1748 |
$xpath = new DOMXpath($doc); |
|
1749 |
// Check for both h-feed and h-entry, as both a feed with no entries |
|
1750 |
// and a list of entries without an h-feed wrapper are both valid. |
|
1751 |
$query = '//*[contains(concat(" ", @class, " "), " h-feed ") or '. |
|
1752 |
'contains(concat(" ", @class, " "), " h-entry ")]'; |
|
1753 |
$result = $xpath->query($query); |
|
1754 |
$microformats = $result->length !== 0; |
|
1755 |
} |
|
1756 |
// Now also do feed discovery, but if microformats were found don't |
|
1757 |
// overwrite the current value of file. |
|
1758 |
$discovered = $locate->find($this->autodiscovery, |
|
1759 |
$this->all_discovered_feeds); |
|
1760 |
if ($microformats) |
|
0 | 1761 |
{ |
16 | 1762 |
if ($hub = $locate->get_rel_link('hub')) |
1763 |
{ |
|
1764 |
$self = $locate->get_rel_link('self'); |
|
1765 |
$this->store_links($file, $hub, $self); |
|
1766 |
} |
|
1767 |
// Push the current file onto all_discovered feeds so the user can |
|
1768 |
// be shown this as one of the options. |
|
1769 |
if (isset($this->all_discovered_feeds)) { |
|
1770 |
$this->all_discovered_feeds[] = $file; |
|
1771 |
} |
|
1772 |
} |
|
1773 |
else |
|
1774 |
{ |
|
1775 |
if ($discovered) |
|
1776 |
{ |
|
1777 |
$file = $discovered; |
|
1778 |
} |
|
1779 |
else |
|
1780 |
{ |
|
1781 |
// We need to unset this so that if SimplePie::set_file() has |
|
1782 |
// been called that object is untouched |
|
1783 |
unset($file); |
|
1784 |
$this->error = "A feed could not be found at `$this->feed_url`; the status code is `$copyStatusCode` and content-type is `$copyContentType`"; |
|
1785 |
$this->registry->call('Misc', 'error', array($this->error, E_USER_NOTICE, __FILE__, __LINE__)); |
|
1786 |
return false; |
|
1787 |
} |
|
0 | 1788 |
} |
1789 |
} |
|
1790 |
catch (SimplePie_Exception $e) |
|
1791 |
{ |
|
16 | 1792 |
// We need to unset this so that if SimplePie::set_file() has been called that object is untouched |
1793 |
unset($file); |
|
0 | 1794 |
// This is usually because DOMDocument doesn't exist |
1795 |
$this->error = $e->getMessage(); |
|
1796 |
$this->registry->call('Misc', 'error', array($this->error, E_USER_NOTICE, $e->getFile(), $e->getLine())); |
|
1797 |
return false; |
|
1798 |
} |
|
1799 |
if ($cache) |
|
1800 |
{ |
|
1801 |
$this->data = array('url' => $this->feed_url, 'feed_url' => $file->url, 'build' => SIMPLEPIE_BUILD); |
|
1802 |
if (!$cache->save($this)) |
|
1803 |
{ |
|
16 | 1804 |
trigger_error("$this->cache_location is not writable. Make sure you've set the correct relative or absolute path, and that the location is server-writable.", E_USER_WARNING); |
0 | 1805 |
} |
1806 |
$cache = $this->registry->call('Cache', 'get_handler', array($this->cache_location, call_user_func($this->cache_name_function, $file->url), 'spc')); |
|
1807 |
} |
|
1808 |
} |
|
16 | 1809 |
$this->feed_url = $file->url; |
0 | 1810 |
$locate = null; |
1811 |
} |
|
1812 |
||
1813 |
$this->raw_data = $file->body; |
|
16 | 1814 |
$this->permanent_url = $file->permanent_url; |
0 | 1815 |
$headers = $file->headers; |
1816 |
$sniffer = $this->registry->create('Content_Type_Sniffer', array(&$file)); |
|
1817 |
$sniffed = $sniffer->get_type(); |
|
1818 |
||
1819 |
return array($headers, $sniffed); |
|
1820 |
} |
|
1821 |
||
1822 |
/** |
|
19 | 1823 |
* Get the error message for the occurred error |
0 | 1824 |
* |
1825 |
* @return string|array Error message, or array of messages for multifeeds |
|
1826 |
*/ |
|
1827 |
public function error() |
|
1828 |
{ |
|
1829 |
return $this->error; |
|
1830 |
} |
|
1831 |
||
1832 |
/** |
|
19 | 1833 |
* Get the last HTTP status code |
1834 |
* |
|
1835 |
* @return int Status code |
|
1836 |
*/ |
|
1837 |
public function status_code() |
|
1838 |
{ |
|
1839 |
return $this->status_code; |
|
1840 |
} |
|
1841 |
||
1842 |
/** |
|
0 | 1843 |
* Get the raw XML |
1844 |
* |
|
1845 |
* This is the same as the old `$feed->enable_xml_dump(true)`, but returns |
|
1846 |
* the data instead of printing it. |
|
1847 |
* |
|
1848 |
* @return string|boolean Raw XML data, false if the cache is used |
|
1849 |
*/ |
|
1850 |
public function get_raw_data() |
|
1851 |
{ |
|
1852 |
return $this->raw_data; |
|
1853 |
} |
|
1854 |
||
1855 |
/** |
|
1856 |
* Get the character encoding used for output |
|
1857 |
* |
|
1858 |
* @since Preview Release |
|
1859 |
* @return string |
|
1860 |
*/ |
|
1861 |
public function get_encoding() |
|
1862 |
{ |
|
1863 |
return $this->sanitize->output_encoding; |
|
1864 |
} |
|
1865 |
||
1866 |
/** |
|
1867 |
* Send the content-type header with correct encoding |
|
1868 |
* |
|
1869 |
* This method ensures that the SimplePie-enabled page is being served with |
|
1870 |
* the correct {@link http://www.iana.org/assignments/media-types/ mime-type} |
|
1871 |
* and character encoding HTTP headers (character encoding determined by the |
|
1872 |
* {@see set_output_encoding} config option). |
|
1873 |
* |
|
1874 |
* This won't work properly if any content or whitespace has already been |
|
1875 |
* sent to the browser, because it relies on PHP's |
|
1876 |
* {@link http://php.net/header header()} function, and these are the |
|
1877 |
* circumstances under which the function works. |
|
1878 |
* |
|
1879 |
* Because it's setting these settings for the entire page (as is the nature |
|
1880 |
* of HTTP headers), this should only be used once per page (again, at the |
|
1881 |
* top). |
|
1882 |
* |
|
1883 |
* @param string $mime MIME type to serve the page as |
|
1884 |
*/ |
|
1885 |
public function handle_content_type($mime = 'text/html') |
|
1886 |
{ |
|
1887 |
if (!headers_sent()) |
|
1888 |
{ |
|
1889 |
$header = "Content-type: $mime;"; |
|
1890 |
if ($this->get_encoding()) |
|
1891 |
{ |
|
1892 |
$header .= ' charset=' . $this->get_encoding(); |
|
1893 |
} |
|
1894 |
else |
|
1895 |
{ |
|
1896 |
$header .= ' charset=UTF-8'; |
|
1897 |
} |
|
1898 |
header($header); |
|
1899 |
} |
|
1900 |
} |
|
1901 |
||
1902 |
/** |
|
1903 |
* Get the type of the feed |
|
1904 |
* |
|
1905 |
* This returns a SIMPLEPIE_TYPE_* constant, which can be tested against |
|
1906 |
* using {@link http://php.net/language.operators.bitwise bitwise operators} |
|
1907 |
* |
|
1908 |
* @since 0.8 (usage changed to using constants in 1.0) |
|
1909 |
* @see SIMPLEPIE_TYPE_NONE Unknown. |
|
1910 |
* @see SIMPLEPIE_TYPE_RSS_090 RSS 0.90. |
|
1911 |
* @see SIMPLEPIE_TYPE_RSS_091_NETSCAPE RSS 0.91 (Netscape). |
|
1912 |
* @see SIMPLEPIE_TYPE_RSS_091_USERLAND RSS 0.91 (Userland). |
|
1913 |
* @see SIMPLEPIE_TYPE_RSS_091 RSS 0.91. |
|
1914 |
* @see SIMPLEPIE_TYPE_RSS_092 RSS 0.92. |
|
1915 |
* @see SIMPLEPIE_TYPE_RSS_093 RSS 0.93. |
|
1916 |
* @see SIMPLEPIE_TYPE_RSS_094 RSS 0.94. |
|
1917 |
* @see SIMPLEPIE_TYPE_RSS_10 RSS 1.0. |
|
1918 |
* @see SIMPLEPIE_TYPE_RSS_20 RSS 2.0.x. |
|
1919 |
* @see SIMPLEPIE_TYPE_RSS_RDF RDF-based RSS. |
|
1920 |
* @see SIMPLEPIE_TYPE_RSS_SYNDICATION Non-RDF-based RSS (truly intended as syndication format). |
|
1921 |
* @see SIMPLEPIE_TYPE_RSS_ALL Any version of RSS. |
|
1922 |
* @see SIMPLEPIE_TYPE_ATOM_03 Atom 0.3. |
|
1923 |
* @see SIMPLEPIE_TYPE_ATOM_10 Atom 1.0. |
|
1924 |
* @see SIMPLEPIE_TYPE_ATOM_ALL Any version of Atom. |
|
1925 |
* @see SIMPLEPIE_TYPE_ALL Any known/supported feed type. |
|
1926 |
* @return int SIMPLEPIE_TYPE_* constant |
|
1927 |
*/ |
|
1928 |
public function get_type() |
|
1929 |
{ |
|
1930 |
if (!isset($this->data['type'])) |
|
1931 |
{ |
|
1932 |
$this->data['type'] = SIMPLEPIE_TYPE_ALL; |
|
1933 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['feed'])) |
|
1934 |
{ |
|
1935 |
$this->data['type'] &= SIMPLEPIE_TYPE_ATOM_10; |
|
1936 |
} |
|
1937 |
elseif (isset($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['feed'])) |
|
1938 |
{ |
|
1939 |
$this->data['type'] &= SIMPLEPIE_TYPE_ATOM_03; |
|
1940 |
} |
|
1941 |
elseif (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'])) |
|
1942 |
{ |
|
1943 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_10]['channel']) |
|
1944 |
|| isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_10]['image']) |
|
1945 |
|| isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_10]['item']) |
|
1946 |
|| isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_10]['textinput'])) |
|
1947 |
{ |
|
1948 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_10; |
|
1949 |
} |
|
1950 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_090]['channel']) |
|
1951 |
|| isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_090]['image']) |
|
1952 |
|| isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_090]['item']) |
|
1953 |
|| isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_090]['textinput'])) |
|
1954 |
{ |
|
1955 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_090; |
|
1956 |
} |
|
1957 |
} |
|
1958 |
elseif (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RSS_20]['rss'])) |
|
1959 |
{ |
|
1960 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_ALL; |
|
1961 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RSS_20]['rss'][0]['attribs']['']['version'])) |
|
1962 |
{ |
|
1963 |
switch (trim($this->data['child'][SIMPLEPIE_NAMESPACE_RSS_20]['rss'][0]['attribs']['']['version'])) |
|
1964 |
{ |
|
1965 |
case '0.91': |
|
1966 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_091; |
|
1967 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RSS_20]['rss'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_20]['skiphours']['hour'][0]['data'])) |
|
1968 |
{ |
|
1969 |
switch (trim($this->data['child'][SIMPLEPIE_NAMESPACE_RSS_20]['rss'][0]['child'][SIMPLEPIE_NAMESPACE_RSS_20]['skiphours']['hour'][0]['data'])) |
|
1970 |
{ |
|
1971 |
case '0': |
|
1972 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_091_NETSCAPE; |
|
1973 |
break; |
|
1974 |
||
1975 |
case '24': |
|
1976 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_091_USERLAND; |
|
1977 |
break; |
|
1978 |
} |
|
1979 |
} |
|
1980 |
break; |
|
1981 |
||
1982 |
case '0.92': |
|
1983 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_092; |
|
1984 |
break; |
|
1985 |
||
1986 |
case '0.93': |
|
1987 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_093; |
|
1988 |
break; |
|
1989 |
||
1990 |
case '0.94': |
|
1991 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_094; |
|
1992 |
break; |
|
1993 |
||
1994 |
case '2.0': |
|
1995 |
$this->data['type'] &= SIMPLEPIE_TYPE_RSS_20; |
|
1996 |
break; |
|
1997 |
} |
|
1998 |
} |
|
1999 |
} |
|
2000 |
else |
|
2001 |
{ |
|
2002 |
$this->data['type'] = SIMPLEPIE_TYPE_NONE; |
|
2003 |
} |
|
2004 |
} |
|
2005 |
return $this->data['type']; |
|
2006 |
} |
|
2007 |
||
2008 |
/** |
|
2009 |
* Get the URL for the feed |
|
2010 |
* |
|
16 | 2011 |
* When the 'permanent' mode is enabled, returns the original feed URL, |
2012 |
* except in the case of an `HTTP 301 Moved Permanently` status response, |
|
2013 |
* in which case the location of the first redirection is returned. |
|
2014 |
* |
|
2015 |
* When the 'permanent' mode is disabled (default), |
|
2016 |
* may or may not be different from the URL passed to {@see set_feed_url()}, |
|
2017 |
* depending on whether auto-discovery was used, and whether there were |
|
2018 |
* any redirects along the way. |
|
0 | 2019 |
* |
2020 |
* @since Preview Release (previously called `get_feed_url()` since SimplePie 0.8.) |
|
16 | 2021 |
* @todo Support <itunes:new-feed-url> |
0 | 2022 |
* @todo Also, |atom:link|@rel=self |
16 | 2023 |
* @param bool $permanent Permanent mode to return only the original URL or the first redirection |
2024 |
* iff it is a 301 redirection |
|
0 | 2025 |
* @return string|null |
2026 |
*/ |
|
16 | 2027 |
public function subscribe_url($permanent = false) |
0 | 2028 |
{ |
16 | 2029 |
if ($permanent) |
0 | 2030 |
{ |
16 | 2031 |
if ($this->permanent_url !== null) |
2032 |
{ |
|
2033 |
// sanitize encodes ampersands which are required when used in a url. |
|
2034 |
return str_replace('&', '&', |
|
2035 |
$this->sanitize($this->permanent_url, |
|
2036 |
SIMPLEPIE_CONSTRUCT_IRI)); |
|
2037 |
} |
|
0 | 2038 |
} |
2039 |
else |
|
2040 |
{ |
|
16 | 2041 |
if ($this->feed_url !== null) |
2042 |
{ |
|
2043 |
return str_replace('&', '&', |
|
2044 |
$this->sanitize($this->feed_url, |
|
2045 |
SIMPLEPIE_CONSTRUCT_IRI)); |
|
2046 |
} |
|
0 | 2047 |
} |
16 | 2048 |
return null; |
0 | 2049 |
} |
2050 |
||
2051 |
/** |
|
2052 |
* Get data for an feed-level element |
|
2053 |
* |
|
2054 |
* This method allows you to get access to ANY element/attribute that is a |
|
2055 |
* sub-element of the opening feed tag. |
|
2056 |
* |
|
2057 |
* The return value is an indexed array of elements matching the given |
|
2058 |
* namespace and tag name. Each element has `attribs`, `data` and `child` |
|
2059 |
* subkeys. For `attribs` and `child`, these contain namespace subkeys. |
|
2060 |
* `attribs` then has one level of associative name => value data (where |
|
2061 |
* `value` is a string) after the namespace. `child` has tag-indexed keys |
|
2062 |
* after the namespace, each member of which is an indexed array matching |
|
2063 |
* this same format. |
|
2064 |
* |
|
2065 |
* For example: |
|
2066 |
* <pre> |
|
2067 |
* // This is probably a bad example because we already support |
|
2068 |
* // <media:content> natively, but it shows you how to parse through |
|
2069 |
* // the nodes. |
|
2070 |
* $group = $item->get_item_tags(SIMPLEPIE_NAMESPACE_MEDIARSS, 'group'); |
|
2071 |
* $content = $group[0]['child'][SIMPLEPIE_NAMESPACE_MEDIARSS]['content']; |
|
2072 |
* $file = $content[0]['attribs']['']['url']; |
|
2073 |
* echo $file; |
|
2074 |
* </pre> |
|
2075 |
* |
|
2076 |
* @since 1.0 |
|
2077 |
* @see http://simplepie.org/wiki/faq/supported_xml_namespaces |
|
2078 |
* @param string $namespace The URL of the XML namespace of the elements you're trying to access |
|
2079 |
* @param string $tag Tag name |
|
2080 |
* @return array |
|
2081 |
*/ |
|
2082 |
public function get_feed_tags($namespace, $tag) |
|
2083 |
{ |
|
2084 |
$type = $this->get_type(); |
|
2085 |
if ($type & SIMPLEPIE_TYPE_ATOM_10) |
|
2086 |
{ |
|
2087 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['feed'][0]['child'][$namespace][$tag])) |
|
2088 |
{ |
|
2089 |
return $this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['feed'][0]['child'][$namespace][$tag]; |
|
2090 |
} |
|
2091 |
} |
|
2092 |
if ($type & SIMPLEPIE_TYPE_ATOM_03) |
|
2093 |
{ |
|
2094 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['feed'][0]['child'][$namespace][$tag])) |
|
2095 |
{ |
|
2096 |
return $this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['feed'][0]['child'][$namespace][$tag]; |
|
2097 |
} |
|
2098 |
} |
|
2099 |
if ($type & SIMPLEPIE_TYPE_RSS_RDF) |
|
2100 |
{ |
|
2101 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][$namespace][$tag])) |
|
2102 |
{ |
|
2103 |
return $this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['child'][$namespace][$tag]; |
|
2104 |
} |
|
2105 |
} |
|
2106 |
if ($type & SIMPLEPIE_TYPE_RSS_SYNDICATION) |
|
2107 |
{ |
|
2108 |
if (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RSS_20]['rss'][0]['child'][$namespace][$tag])) |
|
2109 |
{ |
|
2110 |
return $this->data['child'][SIMPLEPIE_NAMESPACE_RSS_20]['rss'][0]['child'][$namespace][$tag]; |
|
2111 |
} |
|
2112 |
} |
|
2113 |
return null; |
|
2114 |
} |
|
2115 |
||
2116 |
/** |
|
2117 |
* Get data for an channel-level element |
|
2118 |
* |
|
2119 |
* This method allows you to get access to ANY element/attribute in the |
|
2120 |
* channel/header section of the feed. |
|
2121 |
* |
|
2122 |
* See {@see SimplePie::get_feed_tags()} for a description of the return value |
|
2123 |
* |
|
2124 |
* @since 1.0 |
|
2125 |
* @see http://simplepie.org/wiki/faq/supported_xml_namespaces |
|
2126 |
* @param string $namespace The URL of the XML namespace of the elements you're trying to access |
|
2127 |
* @param string $tag Tag name |
|
2128 |
* @return array |
|
2129 |
*/ |
|
2130 |
public function get_channel_tags($namespace, $tag) |
|
2131 |
{ |
|
2132 |
$type = $this->get_type(); |
|
2133 |
if ($type & SIMPLEPIE_TYPE_ATOM_ALL) |
|
2134 |
{ |
|
2135 |
if ($return = $this->get_feed_tags($namespace, $tag)) |
|
2136 |
{ |
|
2137 |
return $return; |
|
2138 |
} |
|
2139 |
} |
|
2140 |
if ($type & SIMPLEPIE_TYPE_RSS_10) |
|
2141 |
{ |
|
2142 |
if ($channel = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'channel')) |
|
2143 |
{ |
|
2144 |
if (isset($channel[0]['child'][$namespace][$tag])) |
|
2145 |
{ |
|
2146 |
return $channel[0]['child'][$namespace][$tag]; |
|
2147 |
} |
|
2148 |
} |
|
2149 |
} |
|
2150 |
if ($type & SIMPLEPIE_TYPE_RSS_090) |
|
2151 |
{ |
|
2152 |
if ($channel = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'channel')) |
|
2153 |
{ |
|
2154 |
if (isset($channel[0]['child'][$namespace][$tag])) |
|
2155 |
{ |
|
2156 |
return $channel[0]['child'][$namespace][$tag]; |
|
2157 |
} |
|
2158 |
} |
|
2159 |
} |
|
2160 |
if ($type & SIMPLEPIE_TYPE_RSS_SYNDICATION) |
|
2161 |
{ |
|
2162 |
if ($channel = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'channel')) |
|
2163 |
{ |
|
2164 |
if (isset($channel[0]['child'][$namespace][$tag])) |
|
2165 |
{ |
|
2166 |
return $channel[0]['child'][$namespace][$tag]; |
|
2167 |
} |
|
2168 |
} |
|
2169 |
} |
|
2170 |
return null; |
|
2171 |
} |
|
2172 |
||
2173 |
/** |
|
2174 |
* Get data for an channel-level element |
|
2175 |
* |
|
2176 |
* This method allows you to get access to ANY element/attribute in the |
|
2177 |
* image/logo section of the feed. |
|
2178 |
* |
|
2179 |
* See {@see SimplePie::get_feed_tags()} for a description of the return value |
|
2180 |
* |
|
2181 |
* @since 1.0 |
|
2182 |
* @see http://simplepie.org/wiki/faq/supported_xml_namespaces |
|
2183 |
* @param string $namespace The URL of the XML namespace of the elements you're trying to access |
|
2184 |
* @param string $tag Tag name |
|
2185 |
* @return array |
|
2186 |
*/ |
|
2187 |
public function get_image_tags($namespace, $tag) |
|
2188 |
{ |
|
2189 |
$type = $this->get_type(); |
|
2190 |
if ($type & SIMPLEPIE_TYPE_RSS_10) |
|
2191 |
{ |
|
2192 |
if ($image = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'image')) |
|
2193 |
{ |
|
2194 |
if (isset($image[0]['child'][$namespace][$tag])) |
|
2195 |
{ |
|
2196 |
return $image[0]['child'][$namespace][$tag]; |
|
2197 |
} |
|
2198 |
} |
|
2199 |
} |
|
2200 |
if ($type & SIMPLEPIE_TYPE_RSS_090) |
|
2201 |
{ |
|
2202 |
if ($image = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'image')) |
|
2203 |
{ |
|
2204 |
if (isset($image[0]['child'][$namespace][$tag])) |
|
2205 |
{ |
|
2206 |
return $image[0]['child'][$namespace][$tag]; |
|
2207 |
} |
|
2208 |
} |
|
2209 |
} |
|
2210 |
if ($type & SIMPLEPIE_TYPE_RSS_SYNDICATION) |
|
2211 |
{ |
|
2212 |
if ($image = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'image')) |
|
2213 |
{ |
|
2214 |
if (isset($image[0]['child'][$namespace][$tag])) |
|
2215 |
{ |
|
2216 |
return $image[0]['child'][$namespace][$tag]; |
|
2217 |
} |
|
2218 |
} |
|
2219 |
} |
|
2220 |
return null; |
|
2221 |
} |
|
2222 |
||
2223 |
/** |
|
2224 |
* Get the base URL value from the feed |
|
2225 |
* |
|
2226 |
* Uses `<xml:base>` if available, otherwise uses the first link in the |
|
2227 |
* feed, or failing that, the URL of the feed itself. |
|
2228 |
* |
|
2229 |
* @see get_link |
|
2230 |
* @see subscribe_url |
|
2231 |
* |
|
2232 |
* @param array $element |
|
2233 |
* @return string |
|
2234 |
*/ |
|
2235 |
public function get_base($element = array()) |
|
2236 |
{ |
|
2237 |
if (!($this->get_type() & SIMPLEPIE_TYPE_RSS_SYNDICATION) && !empty($element['xml_base_explicit']) && isset($element['xml_base'])) |
|
2238 |
{ |
|
2239 |
return $element['xml_base']; |
|
2240 |
} |
|
2241 |
elseif ($this->get_link() !== null) |
|
2242 |
{ |
|
2243 |
return $this->get_link(); |
|
2244 |
} |
|
16 | 2245 |
|
2246 |
return $this->subscribe_url(); |
|
0 | 2247 |
} |
2248 |
||
2249 |
/** |
|
2250 |
* Sanitize feed data |
|
2251 |
* |
|
2252 |
* @access private |
|
2253 |
* @see SimplePie_Sanitize::sanitize() |
|
2254 |
* @param string $data Data to sanitize |
|
2255 |
* @param int $type One of the SIMPLEPIE_CONSTRUCT_* constants |
|
2256 |
* @param string $base Base URL to resolve URLs against |
|
2257 |
* @return string Sanitized data |
|
2258 |
*/ |
|
2259 |
public function sanitize($data, $type, $base = '') |
|
2260 |
{ |
|
16 | 2261 |
try |
2262 |
{ |
|
2263 |
return $this->sanitize->sanitize($data, $type, $base); |
|
2264 |
} |
|
2265 |
catch (SimplePie_Exception $e) |
|
2266 |
{ |
|
2267 |
if (!$this->enable_exceptions) |
|
2268 |
{ |
|
2269 |
$this->error = $e->getMessage(); |
|
2270 |
$this->registry->call('Misc', 'error', array($this->error, E_USER_WARNING, $e->getFile(), $e->getLine())); |
|
2271 |
return ''; |
|
2272 |
} |
|
2273 |
||
2274 |
throw $e; |
|
2275 |
} |
|
0 | 2276 |
} |
2277 |
||
2278 |
/** |
|
2279 |
* Get the title of the feed |
|
2280 |
* |
|
2281 |
* Uses `<atom:title>`, `<title>` or `<dc:title>` |
|
2282 |
* |
|
2283 |
* @since 1.0 (previously called `get_feed_title` since 0.8) |
|
2284 |
* @return string|null |
|
2285 |
*/ |
|
2286 |
public function get_title() |
|
2287 |
{ |
|
2288 |
if ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'title')) |
|
2289 |
{ |
|
2290 |
return $this->sanitize($return[0]['data'], $this->registry->call('Misc', 'atom_10_construct_type', array($return[0]['attribs'])), $this->get_base($return[0])); |
|
2291 |
} |
|
2292 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_03, 'title')) |
|
2293 |
{ |
|
2294 |
return $this->sanitize($return[0]['data'], $this->registry->call('Misc', 'atom_03_construct_type', array($return[0]['attribs'])), $this->get_base($return[0])); |
|
2295 |
} |
|
2296 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'title')) |
|
2297 |
{ |
|
2298 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_MAYBE_HTML, $this->get_base($return[0])); |
|
2299 |
} |
|
2300 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'title')) |
|
2301 |
{ |
|
2302 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_MAYBE_HTML, $this->get_base($return[0])); |
|
2303 |
} |
|
2304 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'title')) |
|
2305 |
{ |
|
2306 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_MAYBE_HTML, $this->get_base($return[0])); |
|
2307 |
} |
|
2308 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_11, 'title')) |
|
2309 |
{ |
|
2310 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2311 |
} |
|
2312 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_10, 'title')) |
|
2313 |
{ |
|
2314 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2315 |
} |
|
16 | 2316 |
|
2317 |
return null; |
|
0 | 2318 |
} |
2319 |
||
2320 |
/** |
|
2321 |
* Get a category for the feed |
|
2322 |
* |
|
2323 |
* @since Unknown |
|
16 | 2324 |
* @param int $key The category that you want to return. Remember that arrays begin with 0, not 1 |
0 | 2325 |
* @return SimplePie_Category|null |
2326 |
*/ |
|
2327 |
public function get_category($key = 0) |
|
2328 |
{ |
|
2329 |
$categories = $this->get_categories(); |
|
2330 |
if (isset($categories[$key])) |
|
2331 |
{ |
|
2332 |
return $categories[$key]; |
|
2333 |
} |
|
16 | 2334 |
|
2335 |
return null; |
|
0 | 2336 |
} |
2337 |
||
2338 |
/** |
|
2339 |
* Get all categories for the feed |
|
2340 |
* |
|
2341 |
* Uses `<atom:category>`, `<category>` or `<dc:subject>` |
|
2342 |
* |
|
2343 |
* @since Unknown |
|
2344 |
* @return array|null List of {@see SimplePie_Category} objects |
|
2345 |
*/ |
|
2346 |
public function get_categories() |
|
2347 |
{ |
|
2348 |
$categories = array(); |
|
2349 |
||
2350 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'category') as $category) |
|
2351 |
{ |
|
2352 |
$term = null; |
|
2353 |
$scheme = null; |
|
2354 |
$label = null; |
|
2355 |
if (isset($category['attribs']['']['term'])) |
|
2356 |
{ |
|
2357 |
$term = $this->sanitize($category['attribs']['']['term'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2358 |
} |
|
2359 |
if (isset($category['attribs']['']['scheme'])) |
|
2360 |
{ |
|
2361 |
$scheme = $this->sanitize($category['attribs']['']['scheme'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2362 |
} |
|
2363 |
if (isset($category['attribs']['']['label'])) |
|
2364 |
{ |
|
2365 |
$label = $this->sanitize($category['attribs']['']['label'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2366 |
} |
|
2367 |
$categories[] = $this->registry->create('Category', array($term, $scheme, $label)); |
|
2368 |
} |
|
2369 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'category') as $category) |
|
2370 |
{ |
|
2371 |
// This is really the label, but keep this as the term also for BC. |
|
2372 |
// Label will also work on retrieving because that falls back to term. |
|
2373 |
$term = $this->sanitize($category['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2374 |
if (isset($category['attribs']['']['domain'])) |
|
2375 |
{ |
|
2376 |
$scheme = $this->sanitize($category['attribs']['']['domain'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2377 |
} |
|
2378 |
else |
|
2379 |
{ |
|
2380 |
$scheme = null; |
|
2381 |
} |
|
2382 |
$categories[] = $this->registry->create('Category', array($term, $scheme, null)); |
|
2383 |
} |
|
2384 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_11, 'subject') as $category) |
|
2385 |
{ |
|
2386 |
$categories[] = $this->registry->create('Category', array($this->sanitize($category['data'], SIMPLEPIE_CONSTRUCT_TEXT), null, null)); |
|
2387 |
} |
|
2388 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_10, 'subject') as $category) |
|
2389 |
{ |
|
2390 |
$categories[] = $this->registry->create('Category', array($this->sanitize($category['data'], SIMPLEPIE_CONSTRUCT_TEXT), null, null)); |
|
2391 |
} |
|
2392 |
||
2393 |
if (!empty($categories)) |
|
2394 |
{ |
|
2395 |
return array_unique($categories); |
|
2396 |
} |
|
16 | 2397 |
|
2398 |
return null; |
|
0 | 2399 |
} |
2400 |
||
2401 |
/** |
|
2402 |
* Get an author for the feed |
|
2403 |
* |
|
2404 |
* @since 1.1 |
|
16 | 2405 |
* @param int $key The author that you want to return. Remember that arrays begin with 0, not 1 |
0 | 2406 |
* @return SimplePie_Author|null |
2407 |
*/ |
|
2408 |
public function get_author($key = 0) |
|
2409 |
{ |
|
2410 |
$authors = $this->get_authors(); |
|
2411 |
if (isset($authors[$key])) |
|
2412 |
{ |
|
2413 |
return $authors[$key]; |
|
2414 |
} |
|
16 | 2415 |
|
2416 |
return null; |
|
0 | 2417 |
} |
2418 |
||
2419 |
/** |
|
2420 |
* Get all authors for the feed |
|
2421 |
* |
|
2422 |
* Uses `<atom:author>`, `<author>`, `<dc:creator>` or `<itunes:author>` |
|
2423 |
* |
|
2424 |
* @since 1.1 |
|
2425 |
* @return array|null List of {@see SimplePie_Author} objects |
|
2426 |
*/ |
|
2427 |
public function get_authors() |
|
2428 |
{ |
|
2429 |
$authors = array(); |
|
2430 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'author') as $author) |
|
2431 |
{ |
|
2432 |
$name = null; |
|
2433 |
$uri = null; |
|
2434 |
$email = null; |
|
2435 |
if (isset($author['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['name'][0]['data'])) |
|
2436 |
{ |
|
2437 |
$name = $this->sanitize($author['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['name'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2438 |
} |
|
2439 |
if (isset($author['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['uri'][0]['data'])) |
|
2440 |
{ |
|
2441 |
$uri = $this->sanitize($author['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['uri'][0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($author['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['uri'][0])); |
|
2442 |
} |
|
2443 |
if (isset($author['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['email'][0]['data'])) |
|
2444 |
{ |
|
2445 |
$email = $this->sanitize($author['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['email'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2446 |
} |
|
2447 |
if ($name !== null || $email !== null || $uri !== null) |
|
2448 |
{ |
|
2449 |
$authors[] = $this->registry->create('Author', array($name, $uri, $email)); |
|
2450 |
} |
|
2451 |
} |
|
2452 |
if ($author = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_03, 'author')) |
|
2453 |
{ |
|
2454 |
$name = null; |
|
2455 |
$url = null; |
|
2456 |
$email = null; |
|
2457 |
if (isset($author[0]['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['name'][0]['data'])) |
|
2458 |
{ |
|
2459 |
$name = $this->sanitize($author[0]['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['name'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2460 |
} |
|
2461 |
if (isset($author[0]['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['url'][0]['data'])) |
|
2462 |
{ |
|
2463 |
$url = $this->sanitize($author[0]['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['url'][0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($author[0]['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['url'][0])); |
|
2464 |
} |
|
2465 |
if (isset($author[0]['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['email'][0]['data'])) |
|
2466 |
{ |
|
2467 |
$email = $this->sanitize($author[0]['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['email'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2468 |
} |
|
2469 |
if ($name !== null || $email !== null || $url !== null) |
|
2470 |
{ |
|
2471 |
$authors[] = $this->registry->create('Author', array($name, $url, $email)); |
|
2472 |
} |
|
2473 |
} |
|
2474 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_11, 'creator') as $author) |
|
2475 |
{ |
|
2476 |
$authors[] = $this->registry->create('Author', array($this->sanitize($author['data'], SIMPLEPIE_CONSTRUCT_TEXT), null, null)); |
|
2477 |
} |
|
2478 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_10, 'creator') as $author) |
|
2479 |
{ |
|
2480 |
$authors[] = $this->registry->create('Author', array($this->sanitize($author['data'], SIMPLEPIE_CONSTRUCT_TEXT), null, null)); |
|
2481 |
} |
|
2482 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ITUNES, 'author') as $author) |
|
2483 |
{ |
|
2484 |
$authors[] = $this->registry->create('Author', array($this->sanitize($author['data'], SIMPLEPIE_CONSTRUCT_TEXT), null, null)); |
|
2485 |
} |
|
2486 |
||
2487 |
if (!empty($authors)) |
|
2488 |
{ |
|
2489 |
return array_unique($authors); |
|
2490 |
} |
|
16 | 2491 |
|
2492 |
return null; |
|
0 | 2493 |
} |
2494 |
||
2495 |
/** |
|
2496 |
* Get a contributor for the feed |
|
2497 |
* |
|
2498 |
* @since 1.1 |
|
16 | 2499 |
* @param int $key The contrbutor that you want to return. Remember that arrays begin with 0, not 1 |
0 | 2500 |
* @return SimplePie_Author|null |
2501 |
*/ |
|
2502 |
public function get_contributor($key = 0) |
|
2503 |
{ |
|
2504 |
$contributors = $this->get_contributors(); |
|
2505 |
if (isset($contributors[$key])) |
|
2506 |
{ |
|
2507 |
return $contributors[$key]; |
|
2508 |
} |
|
16 | 2509 |
|
2510 |
return null; |
|
0 | 2511 |
} |
2512 |
||
2513 |
/** |
|
2514 |
* Get all contributors for the feed |
|
2515 |
* |
|
2516 |
* Uses `<atom:contributor>` |
|
2517 |
* |
|
2518 |
* @since 1.1 |
|
2519 |
* @return array|null List of {@see SimplePie_Author} objects |
|
2520 |
*/ |
|
2521 |
public function get_contributors() |
|
2522 |
{ |
|
2523 |
$contributors = array(); |
|
2524 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'contributor') as $contributor) |
|
2525 |
{ |
|
2526 |
$name = null; |
|
2527 |
$uri = null; |
|
2528 |
$email = null; |
|
2529 |
if (isset($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['name'][0]['data'])) |
|
2530 |
{ |
|
2531 |
$name = $this->sanitize($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['name'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2532 |
} |
|
2533 |
if (isset($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['uri'][0]['data'])) |
|
2534 |
{ |
|
2535 |
$uri = $this->sanitize($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['uri'][0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['uri'][0])); |
|
2536 |
} |
|
2537 |
if (isset($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['email'][0]['data'])) |
|
2538 |
{ |
|
2539 |
$email = $this->sanitize($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['email'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2540 |
} |
|
2541 |
if ($name !== null || $email !== null || $uri !== null) |
|
2542 |
{ |
|
2543 |
$contributors[] = $this->registry->create('Author', array($name, $uri, $email)); |
|
2544 |
} |
|
2545 |
} |
|
2546 |
foreach ((array) $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_03, 'contributor') as $contributor) |
|
2547 |
{ |
|
2548 |
$name = null; |
|
2549 |
$url = null; |
|
2550 |
$email = null; |
|
2551 |
if (isset($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['name'][0]['data'])) |
|
2552 |
{ |
|
2553 |
$name = $this->sanitize($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['name'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2554 |
} |
|
2555 |
if (isset($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['url'][0]['data'])) |
|
2556 |
{ |
|
2557 |
$url = $this->sanitize($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['url'][0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['url'][0])); |
|
2558 |
} |
|
2559 |
if (isset($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['email'][0]['data'])) |
|
2560 |
{ |
|
2561 |
$email = $this->sanitize($contributor['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['email'][0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2562 |
} |
|
2563 |
if ($name !== null || $email !== null || $url !== null) |
|
2564 |
{ |
|
2565 |
$contributors[] = $this->registry->create('Author', array($name, $url, $email)); |
|
2566 |
} |
|
2567 |
} |
|
2568 |
||
2569 |
if (!empty($contributors)) |
|
2570 |
{ |
|
2571 |
return array_unique($contributors); |
|
2572 |
} |
|
16 | 2573 |
|
2574 |
return null; |
|
0 | 2575 |
} |
2576 |
||
2577 |
/** |
|
2578 |
* Get a single link for the feed |
|
2579 |
* |
|
2580 |
* @since 1.0 (previously called `get_feed_link` since Preview Release, `get_feed_permalink()` since 0.8) |
|
16 | 2581 |
* @param int $key The link that you want to return. Remember that arrays begin with 0, not 1 |
0 | 2582 |
* @param string $rel The relationship of the link to return |
2583 |
* @return string|null Link URL |
|
2584 |
*/ |
|
2585 |
public function get_link($key = 0, $rel = 'alternate') |
|
2586 |
{ |
|
2587 |
$links = $this->get_links($rel); |
|
2588 |
if (isset($links[$key])) |
|
2589 |
{ |
|
2590 |
return $links[$key]; |
|
2591 |
} |
|
16 | 2592 |
|
2593 |
return null; |
|
0 | 2594 |
} |
2595 |
||
2596 |
/** |
|
2597 |
* Get the permalink for the item |
|
2598 |
* |
|
2599 |
* Returns the first link available with a relationship of "alternate". |
|
2600 |
* Identical to {@see get_link()} with key 0 |
|
2601 |
* |
|
2602 |
* @see get_link |
|
2603 |
* @since 1.0 (previously called `get_feed_link` since Preview Release, `get_feed_permalink()` since 0.8) |
|
2604 |
* @internal Added for parity between the parent-level and the item/entry-level. |
|
2605 |
* @return string|null Link URL |
|
2606 |
*/ |
|
2607 |
public function get_permalink() |
|
2608 |
{ |
|
2609 |
return $this->get_link(0); |
|
2610 |
} |
|
2611 |
||
2612 |
/** |
|
2613 |
* Get all links for the feed |
|
2614 |
* |
|
2615 |
* Uses `<atom:link>` or `<link>` |
|
2616 |
* |
|
2617 |
* @since Beta 2 |
|
2618 |
* @param string $rel The relationship of links to return |
|
2619 |
* @return array|null Links found for the feed (strings) |
|
2620 |
*/ |
|
2621 |
public function get_links($rel = 'alternate') |
|
2622 |
{ |
|
2623 |
if (!isset($this->data['links'])) |
|
2624 |
{ |
|
2625 |
$this->data['links'] = array(); |
|
2626 |
if ($links = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'link')) |
|
2627 |
{ |
|
2628 |
foreach ($links as $link) |
|
2629 |
{ |
|
2630 |
if (isset($link['attribs']['']['href'])) |
|
2631 |
{ |
|
2632 |
$link_rel = (isset($link['attribs']['']['rel'])) ? $link['attribs']['']['rel'] : 'alternate'; |
|
2633 |
$this->data['links'][$link_rel][] = $this->sanitize($link['attribs']['']['href'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($link)); |
|
2634 |
} |
|
2635 |
} |
|
2636 |
} |
|
2637 |
if ($links = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_03, 'link')) |
|
2638 |
{ |
|
2639 |
foreach ($links as $link) |
|
2640 |
{ |
|
2641 |
if (isset($link['attribs']['']['href'])) |
|
2642 |
{ |
|
2643 |
$link_rel = (isset($link['attribs']['']['rel'])) ? $link['attribs']['']['rel'] : 'alternate'; |
|
2644 |
$this->data['links'][$link_rel][] = $this->sanitize($link['attribs']['']['href'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($link)); |
|
2645 |
||
2646 |
} |
|
2647 |
} |
|
2648 |
} |
|
2649 |
if ($links = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'link')) |
|
2650 |
{ |
|
2651 |
$this->data['links']['alternate'][] = $this->sanitize($links[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($links[0])); |
|
2652 |
} |
|
2653 |
if ($links = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'link')) |
|
2654 |
{ |
|
2655 |
$this->data['links']['alternate'][] = $this->sanitize($links[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($links[0])); |
|
2656 |
} |
|
2657 |
if ($links = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'link')) |
|
2658 |
{ |
|
2659 |
$this->data['links']['alternate'][] = $this->sanitize($links[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($links[0])); |
|
2660 |
} |
|
2661 |
||
2662 |
$keys = array_keys($this->data['links']); |
|
2663 |
foreach ($keys as $key) |
|
2664 |
{ |
|
2665 |
if ($this->registry->call('Misc', 'is_isegment_nz_nc', array($key))) |
|
2666 |
{ |
|
2667 |
if (isset($this->data['links'][SIMPLEPIE_IANA_LINK_RELATIONS_REGISTRY . $key])) |
|
2668 |
{ |
|
2669 |
$this->data['links'][SIMPLEPIE_IANA_LINK_RELATIONS_REGISTRY . $key] = array_merge($this->data['links'][$key], $this->data['links'][SIMPLEPIE_IANA_LINK_RELATIONS_REGISTRY . $key]); |
|
2670 |
$this->data['links'][$key] =& $this->data['links'][SIMPLEPIE_IANA_LINK_RELATIONS_REGISTRY . $key]; |
|
2671 |
} |
|
2672 |
else |
|
2673 |
{ |
|
2674 |
$this->data['links'][SIMPLEPIE_IANA_LINK_RELATIONS_REGISTRY . $key] =& $this->data['links'][$key]; |
|
2675 |
} |
|
2676 |
} |
|
2677 |
elseif (substr($key, 0, 41) === SIMPLEPIE_IANA_LINK_RELATIONS_REGISTRY) |
|
2678 |
{ |
|
2679 |
$this->data['links'][substr($key, 41)] =& $this->data['links'][$key]; |
|
2680 |
} |
|
2681 |
$this->data['links'][$key] = array_unique($this->data['links'][$key]); |
|
2682 |
} |
|
2683 |
} |
|
2684 |
||
19 | 2685 |
if (isset($this->data['headers']['link'])) |
16 | 2686 |
{ |
19 | 2687 |
$link_headers = $this->data['headers']['link']; |
2688 |
if (is_string($link_headers)) { |
|
2689 |
$link_headers = array($link_headers); |
|
2690 |
} |
|
2691 |
$matches = preg_filter('/<([^>]+)>; rel='.preg_quote($rel).'/', '$1', $link_headers); |
|
2692 |
if (!empty($matches)) { |
|
2693 |
return $matches; |
|
2694 |
} |
|
16 | 2695 |
} |
19 | 2696 |
|
2697 |
if (isset($this->data['links'][$rel])) |
|
0 | 2698 |
{ |
2699 |
return $this->data['links'][$rel]; |
|
2700 |
} |
|
16 | 2701 |
|
2702 |
return null; |
|
0 | 2703 |
} |
2704 |
||
2705 |
public function get_all_discovered_feeds() |
|
2706 |
{ |
|
2707 |
return $this->all_discovered_feeds; |
|
2708 |
} |
|
2709 |
||
2710 |
/** |
|
2711 |
* Get the content for the item |
|
2712 |
* |
|
2713 |
* Uses `<atom:subtitle>`, `<atom:tagline>`, `<description>`, |
|
2714 |
* `<dc:description>`, `<itunes:summary>` or `<itunes:subtitle>` |
|
2715 |
* |
|
2716 |
* @since 1.0 (previously called `get_feed_description()` since 0.8) |
|
2717 |
* @return string|null |
|
2718 |
*/ |
|
2719 |
public function get_description() |
|
2720 |
{ |
|
2721 |
if ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'subtitle')) |
|
2722 |
{ |
|
2723 |
return $this->sanitize($return[0]['data'], $this->registry->call('Misc', 'atom_10_construct_type', array($return[0]['attribs'])), $this->get_base($return[0])); |
|
2724 |
} |
|
2725 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_03, 'tagline')) |
|
2726 |
{ |
|
2727 |
return $this->sanitize($return[0]['data'], $this->registry->call('Misc', 'atom_03_construct_type', array($return[0]['attribs'])), $this->get_base($return[0])); |
|
2728 |
} |
|
2729 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'description')) |
|
2730 |
{ |
|
2731 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_MAYBE_HTML, $this->get_base($return[0])); |
|
2732 |
} |
|
2733 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'description')) |
|
2734 |
{ |
|
2735 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_MAYBE_HTML, $this->get_base($return[0])); |
|
2736 |
} |
|
2737 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'description')) |
|
2738 |
{ |
|
2739 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_HTML, $this->get_base($return[0])); |
|
2740 |
} |
|
2741 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_11, 'description')) |
|
2742 |
{ |
|
2743 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2744 |
} |
|
2745 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_10, 'description')) |
|
2746 |
{ |
|
2747 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2748 |
} |
|
2749 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ITUNES, 'summary')) |
|
2750 |
{ |
|
2751 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_HTML, $this->get_base($return[0])); |
|
2752 |
} |
|
2753 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ITUNES, 'subtitle')) |
|
2754 |
{ |
|
2755 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_HTML, $this->get_base($return[0])); |
|
2756 |
} |
|
16 | 2757 |
|
2758 |
return null; |
|
0 | 2759 |
} |
2760 |
||
2761 |
/** |
|
2762 |
* Get the copyright info for the feed |
|
2763 |
* |
|
2764 |
* Uses `<atom:rights>`, `<atom:copyright>` or `<dc:rights>` |
|
2765 |
* |
|
2766 |
* @since 1.0 (previously called `get_feed_copyright()` since 0.8) |
|
2767 |
* @return string|null |
|
2768 |
*/ |
|
2769 |
public function get_copyright() |
|
2770 |
{ |
|
2771 |
if ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'rights')) |
|
2772 |
{ |
|
2773 |
return $this->sanitize($return[0]['data'], $this->registry->call('Misc', 'atom_10_construct_type', array($return[0]['attribs'])), $this->get_base($return[0])); |
|
2774 |
} |
|
2775 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_03, 'copyright')) |
|
2776 |
{ |
|
2777 |
return $this->sanitize($return[0]['data'], $this->registry->call('Misc', 'atom_03_construct_type', array($return[0]['attribs'])), $this->get_base($return[0])); |
|
2778 |
} |
|
2779 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'copyright')) |
|
2780 |
{ |
|
2781 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2782 |
} |
|
2783 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_11, 'rights')) |
|
2784 |
{ |
|
2785 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2786 |
} |
|
2787 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_10, 'rights')) |
|
2788 |
{ |
|
2789 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2790 |
} |
|
16 | 2791 |
|
2792 |
return null; |
|
0 | 2793 |
} |
2794 |
||
2795 |
/** |
|
2796 |
* Get the language for the feed |
|
2797 |
* |
|
2798 |
* Uses `<language>`, `<dc:language>`, or @xml_lang |
|
2799 |
* |
|
2800 |
* @since 1.0 (previously called `get_feed_language()` since 0.8) |
|
2801 |
* @return string|null |
|
2802 |
*/ |
|
2803 |
public function get_language() |
|
2804 |
{ |
|
2805 |
if ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'language')) |
|
2806 |
{ |
|
2807 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2808 |
} |
|
2809 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_11, 'language')) |
|
2810 |
{ |
|
2811 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2812 |
} |
|
2813 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_DC_10, 'language')) |
|
2814 |
{ |
|
2815 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2816 |
} |
|
2817 |
elseif (isset($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['feed'][0]['xml_lang'])) |
|
2818 |
{ |
|
2819 |
return $this->sanitize($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_10]['feed'][0]['xml_lang'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2820 |
} |
|
2821 |
elseif (isset($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['feed'][0]['xml_lang'])) |
|
2822 |
{ |
|
2823 |
return $this->sanitize($this->data['child'][SIMPLEPIE_NAMESPACE_ATOM_03]['feed'][0]['xml_lang'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2824 |
} |
|
2825 |
elseif (isset($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['xml_lang'])) |
|
2826 |
{ |
|
2827 |
return $this->sanitize($this->data['child'][SIMPLEPIE_NAMESPACE_RDF]['RDF'][0]['xml_lang'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2828 |
} |
|
2829 |
elseif (isset($this->data['headers']['content-language'])) |
|
2830 |
{ |
|
2831 |
return $this->sanitize($this->data['headers']['content-language'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2832 |
} |
|
16 | 2833 |
|
2834 |
return null; |
|
0 | 2835 |
} |
2836 |
||
2837 |
/** |
|
2838 |
* Get the latitude coordinates for the item |
|
2839 |
* |
|
2840 |
* Compatible with the W3C WGS84 Basic Geo and GeoRSS specifications |
|
2841 |
* |
|
2842 |
* Uses `<geo:lat>` or `<georss:point>` |
|
2843 |
* |
|
2844 |
* @since 1.0 |
|
2845 |
* @link http://www.w3.org/2003/01/geo/ W3C WGS84 Basic Geo |
|
2846 |
* @link http://www.georss.org/ GeoRSS |
|
2847 |
* @return string|null |
|
2848 |
*/ |
|
2849 |
public function get_latitude() |
|
2850 |
{ |
|
2851 |
||
2852 |
if ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_W3C_BASIC_GEO, 'lat')) |
|
2853 |
{ |
|
2854 |
return (float) $return[0]['data']; |
|
2855 |
} |
|
2856 |
elseif (($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_GEORSS, 'point')) && preg_match('/^((?:-)?[0-9]+(?:\.[0-9]+)) ((?:-)?[0-9]+(?:\.[0-9]+))$/', trim($return[0]['data']), $match)) |
|
2857 |
{ |
|
2858 |
return (float) $match[1]; |
|
2859 |
} |
|
16 | 2860 |
|
2861 |
return null; |
|
0 | 2862 |
} |
2863 |
||
2864 |
/** |
|
2865 |
* Get the longitude coordinates for the feed |
|
2866 |
* |
|
2867 |
* Compatible with the W3C WGS84 Basic Geo and GeoRSS specifications |
|
2868 |
* |
|
2869 |
* Uses `<geo:long>`, `<geo:lon>` or `<georss:point>` |
|
2870 |
* |
|
2871 |
* @since 1.0 |
|
2872 |
* @link http://www.w3.org/2003/01/geo/ W3C WGS84 Basic Geo |
|
2873 |
* @link http://www.georss.org/ GeoRSS |
|
2874 |
* @return string|null |
|
2875 |
*/ |
|
2876 |
public function get_longitude() |
|
2877 |
{ |
|
2878 |
if ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_W3C_BASIC_GEO, 'long')) |
|
2879 |
{ |
|
2880 |
return (float) $return[0]['data']; |
|
2881 |
} |
|
2882 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_W3C_BASIC_GEO, 'lon')) |
|
2883 |
{ |
|
2884 |
return (float) $return[0]['data']; |
|
2885 |
} |
|
2886 |
elseif (($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_GEORSS, 'point')) && preg_match('/^((?:-)?[0-9]+(?:\.[0-9]+)) ((?:-)?[0-9]+(?:\.[0-9]+))$/', trim($return[0]['data']), $match)) |
|
2887 |
{ |
|
2888 |
return (float) $match[2]; |
|
2889 |
} |
|
16 | 2890 |
|
2891 |
return null; |
|
0 | 2892 |
} |
2893 |
||
2894 |
/** |
|
2895 |
* Get the feed logo's title |
|
2896 |
* |
|
2897 |
* RSS 0.9.0, 1.0 and 2.0 feeds are allowed to have a "feed logo" title. |
|
2898 |
* |
|
2899 |
* Uses `<image><title>` or `<image><dc:title>` |
|
2900 |
* |
|
2901 |
* @return string|null |
|
2902 |
*/ |
|
2903 |
public function get_image_title() |
|
2904 |
{ |
|
2905 |
if ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'title')) |
|
2906 |
{ |
|
2907 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2908 |
} |
|
2909 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'title')) |
|
2910 |
{ |
|
2911 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2912 |
} |
|
2913 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'title')) |
|
2914 |
{ |
|
2915 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2916 |
} |
|
2917 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_DC_11, 'title')) |
|
2918 |
{ |
|
2919 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2920 |
} |
|
2921 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_DC_10, 'title')) |
|
2922 |
{ |
|
2923 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_TEXT); |
|
2924 |
} |
|
16 | 2925 |
|
2926 |
return null; |
|
0 | 2927 |
} |
2928 |
||
2929 |
/** |
|
2930 |
* Get the feed logo's URL |
|
2931 |
* |
|
2932 |
* RSS 0.9.0, 2.0, Atom 1.0, and feeds with iTunes RSS tags are allowed to |
|
2933 |
* have a "feed logo" URL. This points directly to the image itself. |
|
2934 |
* |
|
2935 |
* Uses `<itunes:image>`, `<atom:logo>`, `<atom:icon>`, |
|
2936 |
* `<image><title>` or `<image><dc:title>` |
|
2937 |
* |
|
2938 |
* @return string|null |
|
2939 |
*/ |
|
2940 |
public function get_image_url() |
|
2941 |
{ |
|
2942 |
if ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ITUNES, 'image')) |
|
2943 |
{ |
|
2944 |
return $this->sanitize($return[0]['attribs']['']['href'], SIMPLEPIE_CONSTRUCT_IRI); |
|
2945 |
} |
|
2946 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'logo')) |
|
2947 |
{ |
|
2948 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2949 |
} |
|
2950 |
elseif ($return = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'icon')) |
|
2951 |
{ |
|
2952 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2953 |
} |
|
2954 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'url')) |
|
2955 |
{ |
|
2956 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2957 |
} |
|
2958 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'url')) |
|
2959 |
{ |
|
2960 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2961 |
} |
|
2962 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'url')) |
|
2963 |
{ |
|
2964 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2965 |
} |
|
16 | 2966 |
|
2967 |
return null; |
|
0 | 2968 |
} |
2969 |
||
2970 |
||
2971 |
/** |
|
2972 |
* Get the feed logo's link |
|
2973 |
* |
|
2974 |
* RSS 0.9.0, 1.0 and 2.0 feeds are allowed to have a "feed logo" link. This |
|
2975 |
* points to a human-readable page that the image should link to. |
|
2976 |
* |
|
2977 |
* Uses `<itunes:image>`, `<atom:logo>`, `<atom:icon>`, |
|
2978 |
* `<image><title>` or `<image><dc:title>` |
|
2979 |
* |
|
2980 |
* @return string|null |
|
2981 |
*/ |
|
2982 |
public function get_image_link() |
|
2983 |
{ |
|
2984 |
if ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'link')) |
|
2985 |
{ |
|
2986 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2987 |
} |
|
2988 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'link')) |
|
2989 |
{ |
|
2990 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2991 |
} |
|
2992 |
elseif ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'link')) |
|
2993 |
{ |
|
2994 |
return $this->sanitize($return[0]['data'], SIMPLEPIE_CONSTRUCT_IRI, $this->get_base($return[0])); |
|
2995 |
} |
|
16 | 2996 |
|
2997 |
return null; |
|
0 | 2998 |
} |
2999 |
||
3000 |
/** |
|
3001 |
* Get the feed logo's link |
|
3002 |
* |
|
3003 |
* RSS 2.0 feeds are allowed to have a "feed logo" width. |
|
3004 |
* |
|
3005 |
* Uses `<image><width>` or defaults to 88.0 if no width is specified and |
|
3006 |
* the feed is an RSS 2.0 feed. |
|
3007 |
* |
|
3008 |
* @return int|float|null |
|
3009 |
*/ |
|
3010 |
public function get_image_width() |
|
3011 |
{ |
|
3012 |
if ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'width')) |
|
3013 |
{ |
|
3014 |
return round($return[0]['data']); |
|
3015 |
} |
|
3016 |
elseif ($this->get_type() & SIMPLEPIE_TYPE_RSS_SYNDICATION && $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'url')) |
|
3017 |
{ |
|
3018 |
return 88.0; |
|
3019 |
} |
|
16 | 3020 |
|
3021 |
return null; |
|
0 | 3022 |
} |
3023 |
||
3024 |
/** |
|
3025 |
* Get the feed logo's height |
|
3026 |
* |
|
3027 |
* RSS 2.0 feeds are allowed to have a "feed logo" height. |
|
3028 |
* |
|
3029 |
* Uses `<image><height>` or defaults to 31.0 if no height is specified and |
|
3030 |
* the feed is an RSS 2.0 feed. |
|
3031 |
* |
|
3032 |
* @return int|float|null |
|
3033 |
*/ |
|
3034 |
public function get_image_height() |
|
3035 |
{ |
|
3036 |
if ($return = $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'height')) |
|
3037 |
{ |
|
3038 |
return round($return[0]['data']); |
|
3039 |
} |
|
3040 |
elseif ($this->get_type() & SIMPLEPIE_TYPE_RSS_SYNDICATION && $this->get_image_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'url')) |
|
3041 |
{ |
|
3042 |
return 31.0; |
|
3043 |
} |
|
16 | 3044 |
|
3045 |
return null; |
|
0 | 3046 |
} |
3047 |
||
3048 |
/** |
|
3049 |
* Get the number of items in the feed |
|
3050 |
* |
|
3051 |
* This is well-suited for {@link http://php.net/for for()} loops with |
|
3052 |
* {@see get_item()} |
|
3053 |
* |
|
3054 |
* @param int $max Maximum value to return. 0 for no limit |
|
3055 |
* @return int Number of items in the feed |
|
3056 |
*/ |
|
3057 |
public function get_item_quantity($max = 0) |
|
3058 |
{ |
|
3059 |
$max = (int) $max; |
|
3060 |
$qty = count($this->get_items()); |
|
3061 |
if ($max === 0) |
|
3062 |
{ |
|
3063 |
return $qty; |
|
3064 |
} |
|
16 | 3065 |
|
3066 |
return ($qty > $max) ? $max : $qty; |
|
0 | 3067 |
} |
3068 |
||
3069 |
/** |
|
3070 |
* Get a single item from the feed |
|
3071 |
* |
|
3072 |
* This is better suited for {@link http://php.net/for for()} loops, whereas |
|
3073 |
* {@see get_items()} is better suited for |
|
3074 |
* {@link http://php.net/foreach foreach()} loops. |
|
3075 |
* |
|
3076 |
* @see get_item_quantity() |
|
3077 |
* @since Beta 2 |
|
16 | 3078 |
* @param int $key The item that you want to return. Remember that arrays begin with 0, not 1 |
0 | 3079 |
* @return SimplePie_Item|null |
3080 |
*/ |
|
3081 |
public function get_item($key = 0) |
|
3082 |
{ |
|
3083 |
$items = $this->get_items(); |
|
3084 |
if (isset($items[$key])) |
|
3085 |
{ |
|
3086 |
return $items[$key]; |
|
3087 |
} |
|
16 | 3088 |
|
3089 |
return null; |
|
0 | 3090 |
} |
3091 |
||
3092 |
/** |
|
3093 |
* Get all items from the feed |
|
3094 |
* |
|
3095 |
* This is better suited for {@link http://php.net/for for()} loops, whereas |
|
3096 |
* {@see get_items()} is better suited for |
|
3097 |
* {@link http://php.net/foreach foreach()} loops. |
|
3098 |
* |
|
3099 |
* @see get_item_quantity |
|
3100 |
* @since Beta 2 |
|
3101 |
* @param int $start Index to start at |
|
3102 |
* @param int $end Number of items to return. 0 for all items after `$start` |
|
16 | 3103 |
* @return SimplePie_Item[]|null List of {@see SimplePie_Item} objects |
0 | 3104 |
*/ |
3105 |
public function get_items($start = 0, $end = 0) |
|
3106 |
{ |
|
3107 |
if (!isset($this->data['items'])) |
|
3108 |
{ |
|
3109 |
if (!empty($this->multifeed_objects)) |
|
3110 |
{ |
|
3111 |
$this->data['items'] = SimplePie::merge_items($this->multifeed_objects, $start, $end, $this->item_limit); |
|
16 | 3112 |
if (empty($this->data['items'])) |
0 | 3113 |
{ |
16 | 3114 |
return array(); |
0 | 3115 |
} |
16 | 3116 |
return $this->data['items']; |
3117 |
} |
|
3118 |
$this->data['items'] = array(); |
|
3119 |
if ($items = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'entry')) |
|
3120 |
{ |
|
3121 |
$keys = array_keys($items); |
|
3122 |
foreach ($keys as $key) |
|
0 | 3123 |
{ |
16 | 3124 |
$this->data['items'][] = $this->registry->create('Item', array($this, $items[$key])); |
0 | 3125 |
} |
16 | 3126 |
} |
3127 |
if ($items = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_ATOM_03, 'entry')) |
|
3128 |
{ |
|
3129 |
$keys = array_keys($items); |
|
3130 |
foreach ($keys as $key) |
|
0 | 3131 |
{ |
16 | 3132 |
$this->data['items'][] = $this->registry->create('Item', array($this, $items[$key])); |
0 | 3133 |
} |
16 | 3134 |
} |
3135 |
if ($items = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_RSS_10, 'item')) |
|
3136 |
{ |
|
3137 |
$keys = array_keys($items); |
|
3138 |
foreach ($keys as $key) |
|
0 | 3139 |
{ |
16 | 3140 |
$this->data['items'][] = $this->registry->create('Item', array($this, $items[$key])); |
0 | 3141 |
} |
16 | 3142 |
} |
3143 |
if ($items = $this->get_feed_tags(SIMPLEPIE_NAMESPACE_RSS_090, 'item')) |
|
3144 |
{ |
|
3145 |
$keys = array_keys($items); |
|
3146 |
foreach ($keys as $key) |
|
0 | 3147 |
{ |
16 | 3148 |
$this->data['items'][] = $this->registry->create('Item', array($this, $items[$key])); |
3149 |
} |
|
3150 |
} |
|
3151 |
if ($items = $this->get_channel_tags(SIMPLEPIE_NAMESPACE_RSS_20, 'item')) |
|
3152 |
{ |
|
3153 |
$keys = array_keys($items); |
|
3154 |
foreach ($keys as $key) |
|
3155 |
{ |
|
3156 |
$this->data['items'][] = $this->registry->create('Item', array($this, $items[$key])); |
|
0 | 3157 |
} |
3158 |
} |
|
3159 |
} |
|
3160 |
||
16 | 3161 |
if (empty($this->data['items'])) |
0 | 3162 |
{ |
16 | 3163 |
return array(); |
3164 |
} |
|
3165 |
||
3166 |
if ($this->order_by_date) |
|
3167 |
{ |
|
3168 |
if (!isset($this->data['ordered_items'])) |
|
0 | 3169 |
{ |
16 | 3170 |
$this->data['ordered_items'] = $this->data['items']; |
3171 |
usort($this->data['ordered_items'], array(get_class($this), 'sort_items')); |
|
3172 |
} |
|
3173 |
$items = $this->data['ordered_items']; |
|
0 | 3174 |
} |
3175 |
else |
|
3176 |
{ |
|
16 | 3177 |
$items = $this->data['items']; |
0 | 3178 |
} |
16 | 3179 |
// Slice the data as desired |
3180 |
if ($end === 0) |
|
3181 |
{ |
|
3182 |
return array_slice($items, $start); |
|
3183 |
} |
|
3184 |
||
3185 |
return array_slice($items, $start, $end); |
|
0 | 3186 |
} |
3187 |
||
3188 |
/** |
|
3189 |
* Set the favicon handler |
|
3190 |
* |
|
3191 |
* @deprecated Use your own favicon handling instead |
|
3192 |
*/ |
|
3193 |
public function set_favicon_handler($page = false, $qs = 'i') |
|
3194 |
{ |
|
3195 |
$level = defined('E_USER_DEPRECATED') ? E_USER_DEPRECATED : E_USER_WARNING; |
|
3196 |
trigger_error('Favicon handling has been removed, please use your own handling', $level); |
|
3197 |
return false; |
|
3198 |
} |
|
3199 |
||
3200 |
/** |
|
3201 |
* Get the favicon for the current feed |
|
3202 |
* |
|
3203 |
* @deprecated Use your own favicon handling instead |
|
3204 |
*/ |
|
3205 |
public function get_favicon() |
|
3206 |
{ |
|
3207 |
$level = defined('E_USER_DEPRECATED') ? E_USER_DEPRECATED : E_USER_WARNING; |
|
3208 |
trigger_error('Favicon handling has been removed, please use your own handling', $level); |
|
3209 |
||
3210 |
if (($url = $this->get_link()) !== null) |
|
3211 |
{ |
|
16 | 3212 |
return 'https://www.google.com/s2/favicons?domain=' . urlencode($url); |
0 | 3213 |
} |
3214 |
||
3215 |
return false; |
|
3216 |
} |
|
3217 |
||
3218 |
/** |
|
3219 |
* Magic method handler |
|
3220 |
* |
|
3221 |
* @param string $method Method name |
|
3222 |
* @param array $args Arguments to the method |
|
3223 |
* @return mixed |
|
3224 |
*/ |
|
3225 |
public function __call($method, $args) |
|
3226 |
{ |
|
3227 |
if (strpos($method, 'subscribe_') === 0) |
|
3228 |
{ |
|
3229 |
$level = defined('E_USER_DEPRECATED') ? E_USER_DEPRECATED : E_USER_WARNING; |
|
3230 |
trigger_error('subscribe_*() has been deprecated, implement the callback yourself', $level); |
|
3231 |
return ''; |
|
3232 |
} |
|
3233 |
if ($method === 'enable_xml_dump') |
|
3234 |
{ |
|
3235 |
$level = defined('E_USER_DEPRECATED') ? E_USER_DEPRECATED : E_USER_WARNING; |
|
3236 |
trigger_error('enable_xml_dump() has been deprecated, use get_raw_data() instead', $level); |
|
3237 |
return false; |
|
3238 |
} |
|
3239 |
||
3240 |
$class = get_class($this); |
|
16 | 3241 |
$trace = debug_backtrace(); // phpcs:ignore PHPCompatibility.FunctionUse.ArgumentFunctionsReportCurrentValue.NeedsInspection |
0 | 3242 |
$file = $trace[0]['file']; |
3243 |
$line = $trace[0]['line']; |
|
3244 |
trigger_error("Call to undefined method $class::$method() in $file on line $line", E_USER_ERROR); |
|
3245 |
} |
|
3246 |
||
3247 |
/** |
|
3248 |
* Sorting callback for items |
|
3249 |
* |
|
3250 |
* @access private |
|
3251 |
* @param SimplePie $a |
|
3252 |
* @param SimplePie $b |
|
3253 |
* @return boolean |
|
3254 |
*/ |
|
3255 |
public static function sort_items($a, $b) |
|
3256 |
{ |
|
16 | 3257 |
$a_date = $a->get_date('U'); |
3258 |
$b_date = $b->get_date('U'); |
|
3259 |
if ($a_date && $b_date) { |
|
3260 |
return $a_date > $b_date ? -1 : 1; |
|
3261 |
} |
|
3262 |
// Sort items without dates to the top. |
|
3263 |
if ($a_date) { |
|
3264 |
return 1; |
|
3265 |
} |
|
3266 |
if ($b_date) { |
|
3267 |
return -1; |
|
3268 |
} |
|
3269 |
return 0; |
|
0 | 3270 |
} |
3271 |
||
3272 |
/** |
|
3273 |
* Merge items from several feeds into one |
|
3274 |
* |
|
3275 |
* If you're merging multiple feeds together, they need to all have dates |
|
3276 |
* for the items or else SimplePie will refuse to sort them. |
|
3277 |
* |
|
3278 |
* @link http://simplepie.org/wiki/tutorial/sort_multiple_feeds_by_time_and_date#if_feeds_require_separate_per-feed_settings |
|
3279 |
* @param array $urls List of SimplePie feed objects to merge |
|
3280 |
* @param int $start Starting item |
|
3281 |
* @param int $end Number of items to return |
|
3282 |
* @param int $limit Maximum number of items per feed |
|
3283 |
* @return array |
|
3284 |
*/ |
|
3285 |
public static function merge_items($urls, $start = 0, $end = 0, $limit = 0) |
|
3286 |
{ |
|
3287 |
if (is_array($urls) && sizeof($urls) > 0) |
|
3288 |
{ |
|
3289 |
$items = array(); |
|
3290 |
foreach ($urls as $arg) |
|
3291 |
{ |
|
3292 |
if ($arg instanceof SimplePie) |
|
3293 |
{ |
|
3294 |
$items = array_merge($items, $arg->get_items(0, $limit)); |
|
3295 |
} |
|
3296 |
else |
|
3297 |
{ |
|
3298 |
trigger_error('Arguments must be SimplePie objects', E_USER_WARNING); |
|
3299 |
} |
|
3300 |
} |
|
3301 |
||
16 | 3302 |
usort($items, array(get_class($urls[0]), 'sort_items')); |
0 | 3303 |
|
3304 |
if ($end === 0) |
|
3305 |
{ |
|
3306 |
return array_slice($items, $start); |
|
3307 |
} |
|
16 | 3308 |
|
3309 |
return array_slice($items, $start, $end); |
|
3310 |
} |
|
3311 |
||
3312 |
trigger_error('Cannot merge zero SimplePie objects', E_USER_WARNING); |
|
3313 |
return array(); |
|
3314 |
} |
|
3315 |
||
3316 |
/** |
|
3317 |
* Store PubSubHubbub links as headers |
|
3318 |
* |
|
3319 |
* There is no way to find PuSH links in the body of a microformats feed, |
|
3320 |
* so they are added to the headers when found, to be used later by get_links. |
|
3321 |
* @param SimplePie_File $file |
|
3322 |
* @param string $hub |
|
3323 |
* @param string $self |
|
3324 |
*/ |
|
3325 |
private function store_links(&$file, $hub, $self) { |
|
3326 |
if (isset($file->headers['link']['hub']) || |
|
3327 |
(isset($file->headers['link']) && |
|
3328 |
preg_match('/rel=hub/', $file->headers['link']))) |
|
3329 |
{ |
|
3330 |
return; |
|
3331 |
} |
|
3332 |
||
3333 |
if ($hub) |
|
3334 |
{ |
|
3335 |
if (isset($file->headers['link'])) |
|
3336 |
{ |
|
3337 |
if ($file->headers['link'] !== '') |
|
3338 |
{ |
|
3339 |
$file->headers['link'] = ', '; |
|
3340 |
} |
|
3341 |
} |
|
0 | 3342 |
else |
3343 |
{ |
|
16 | 3344 |
$file->headers['link'] = ''; |
0 | 3345 |
} |
16 | 3346 |
$file->headers['link'] .= '<'.$hub.'>; rel=hub'; |
3347 |
if ($self) |
|
3348 |
{ |
|
3349 |
$file->headers['link'] .= ', <'.$self.'>; rel=self'; |
|
3350 |
} |
|
0 | 3351 |
} |
3352 |
} |
|
3353 |
} |
|
16 | 3354 |
endif; |