forked from GNUsocial/gnu-social
		
	
		
			
	
	
		
			358 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			358 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
|   | <?php | ||
|  | 
 | ||
|  | /** | ||
|  |  * An interface for oEmbed consumption | ||
|  |  * | ||
|  |  * PHP version 5.1.0+ | ||
|  |  * | ||
|  |  * Copyright (c) 2008, Digg.com, Inc. | ||
|  |  *  | ||
|  |  * All rights reserved. | ||
|  |  *  | ||
|  |  * Redistribution and use in source and binary forms, with or without  | ||
|  |  * modification, are permitted provided that the following conditions are met: | ||
|  |  * | ||
|  |  *  - Redistributions of source code must retain the above copyright notice, | ||
|  |  *    this list of conditions and the following disclaimer. | ||
|  |  *  - Redistributions in binary form must reproduce the above copyright notice, | ||
|  |  *    this list of conditions and the following disclaimer in the documentation | ||
|  |  *    and/or other materials provided with the distribution. | ||
|  |  *  - Neither the name of Digg.com, Inc. nor the names of its contributors  | ||
|  |  *    may be used to endorse or promote products derived from this software  | ||
|  |  *    without specific prior written permission. | ||
|  |  * | ||
|  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"  | ||
|  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE  | ||
|  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE  | ||
|  |  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE  | ||
|  |  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR  | ||
|  |  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF  | ||
|  |  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS  | ||
|  |  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN  | ||
|  |  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)  | ||
|  |  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE  | ||
|  |  * POSSIBILITY OF SUCH DAMAGE. | ||
|  |  * | ||
|  |  * @category  Services | ||
|  |  * @package   Services_oEmbed | ||
|  |  * @author    Joe Stump <joe@joestump.net>  | ||
|  |  * @copyright 2008 Digg.com, Inc. | ||
|  |  * @license   http://tinyurl.com/42zef New BSD License | ||
|  |  * @version   SVN: @version@ | ||
|  |  * @link      http://code.google.com/p/digg | ||
|  |  * @link      http://oembed.com | ||
|  |  */ | ||
|  | 
 | ||
|  | require_once 'Validate.php'; | ||
|  | require_once 'Net/URL2.php'; | ||
|  | require_once 'HTTP/Request.php'; | ||
|  | require_once 'Services/oEmbed/Exception.php'; | ||
|  | require_once 'Services/oEmbed/Exception/NoSupport.php'; | ||
|  | require_once 'Services/oEmbed/Object.php'; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Base class for consuming oEmbed objects | ||
|  |  * | ||
|  |  * <code> | ||
|  |  * <?php | ||
|  |  *  | ||
|  |  * require_once 'Services/oEmbed.php'; | ||
|  |  * | ||
|  |  * // The URL that we'd like to find out more information about.
 | ||
|  |  * $url = 'http://flickr.com/photos/joestump/2848795611/'; | ||
|  |  * | ||
|  |  * // The oEmbed API URI. Not all providers support discovery yet so we're
 | ||
|  |  * // explicitly providing one here. If one is not provided Services_oEmbed
 | ||
|  |  * // attempts to discover it. If none is found an exception is thrown.
 | ||
|  |  * $oEmbed = new Services_oEmbed($url, array( | ||
|  |  *     Services_oEmbed::OPTION_API => 'http://www.flickr.com/services/oembed/' | ||
|  |  * )); | ||
|  |  * $object = $oEmbed->getObject(); | ||
|  |  * | ||
|  |  * // All of the objects have somewhat sane __toString() methods that allow
 | ||
|  |  * // you to output them directly.
 | ||
|  |  * echo (string)$object; | ||
|  |  *  | ||
|  |  * ?>
 | ||
|  |  * </code>  | ||
|  |  *  | ||
|  |  * @category  Services | ||
|  |  * @package   Services_oEmbed | ||
|  |  * @author    Joe Stump <joe@joestump.net>  | ||
|  |  * @copyright 2008 Digg.com, Inc. | ||
|  |  * @license   http://tinyurl.com/42zef New BSD License | ||
|  |  * @version   Release: @version@ | ||
|  |  * @link      http://code.google.com/p/digg | ||
|  |  * @link      http://oembed.com | ||
|  |  */ | ||
|  | class Services_oEmbed | ||
|  | { | ||
|  |     /** | ||
|  |      * HTTP timeout in seconds | ||
|  |      *  | ||
|  |      * All HTTP requests made by Services_oEmbed will respect this timeout.  | ||
|  |      * This can be passed to {@link Services_oEmbed::setOption()} or to the | ||
|  |      * options parameter in {@link Services_oEmbed::__construct()}. | ||
|  |      *  | ||
|  |      * @var string OPTION_TIMEOUT Timeout in seconds  | ||
|  |      */ | ||
|  |     const OPTION_TIMEOUT = 'http_timeout'; | ||
|  | 
 | ||
|  |     /** | ||
|  |      * HTTP User-Agent  | ||
|  |      * | ||
|  |      * All HTTP requests made by Services_oEmbed will be sent with the  | ||
|  |      * string set by this option. | ||
|  |      * | ||
|  |      * @var string OPTION_USER_AGENT The HTTP User-Agent string | ||
|  |      */ | ||
|  |     const OPTION_USER_AGENT = 'http_user_agent'; | ||
|  | 
 | ||
|  |     /** | ||
|  |      * The API's URI | ||
|  |      * | ||
|  |      * If the API is known ahead of time this option can be used to explicitly | ||
|  |      * set it. If not present then the API is attempted to be discovered  | ||
|  |      * through the auto-discovery mechanism. | ||
|  |      * | ||
|  |      * @var string OPTION_API | ||
|  |      */ | ||
|  |     const OPTION_API = 'oembed_api'; | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Options for oEmbed requests | ||
|  |      * | ||
|  |      * @var array $options The options for making requests | ||
|  |      */ | ||
|  |     protected $options = array( | ||
|  |         self::OPTION_TIMEOUT    => 3, | ||
|  |         self::OPTION_API        => null, | ||
|  |         self::OPTION_USER_AGENT => 'Services_oEmbed 0.1.0' | ||
|  |     ); | ||
|  | 
 | ||
|  |     /** | ||
|  |      * URL of object to get embed information for | ||
|  |      * | ||
|  |      * @var object $url {@link Net_URL2} instance of URL of object | ||
|  |      */ | ||
|  |     protected $url = null; | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Constructor | ||
|  |      * | ||
|  |      * @param string $url     The URL to fetch an oEmbed for | ||
|  |      * @param array  $options A list of options for the oEmbed lookup | ||
|  |      * | ||
|  |      * @throws {@link Services_oEmbed_Exception} if the $url is invalid | ||
|  |      * @throws {@link Services_oEmbed_Exception} when no valid API is found | ||
|  |      * @return void | ||
|  |      */ | ||
|  |     public function __construct($url, array $options = array()) | ||
|  |     { | ||
|  |         if (Validate::uri($url)) { | ||
|  |             $this->url = new Net_URL2($url); | ||
|  |         } else { | ||
|  |             throw new Services_oEmbed_Exception('URL is invalid'); | ||
|  |         } | ||
|  | 
 | ||
|  |         if (count($options)) { | ||
|  |             foreach ($options as $key => $val) { | ||
|  |                 $this->setOption($key, $val); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         if ($this->options[self::OPTION_API] === null) { | ||
|  |             $this->options[self::OPTION_API] = $this->discover(); | ||
|  |         }  | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Set an option for the oEmbed request | ||
|  |      *  | ||
|  |      * @param mixed $option The option name | ||
|  |      * @param mixed $value  The option value | ||
|  |      * | ||
|  |      * @see Services_oEmbed::OPTION_API, Services_oEmbed::OPTION_TIMEOUT | ||
|  |      * @throws {@link Services_oEmbed_Exception} on invalid option | ||
|  |      * @access public | ||
|  |      * @return void | ||
|  |      */ | ||
|  |     public function setOption($option, $value) | ||
|  |     { | ||
|  |         switch ($option) { | ||
|  |         case self::OPTION_API: | ||
|  |         case self::OPTION_TIMEOUT: | ||
|  |             break; | ||
|  |         default: | ||
|  |             throw new Services_oEmbed_Exception( | ||
|  |                 'Invalid option "' . $option . '"' | ||
|  |             ); | ||
|  |         } | ||
|  | 
 | ||
|  |         $func = '_set_' . $option; | ||
|  |         if (method_exists($this, $func)) { | ||
|  |             $this->options[$option] = $this->$func($value); | ||
|  |         } else { | ||
|  |             $this->options[$option] = $value; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Set the API option | ||
|  |      *  | ||
|  |      * @param string $value The API's URI | ||
|  |      * | ||
|  |      * @throws {@link Services_oEmbed_Exception} on invalid API URI | ||
|  |      * @see Validate::uri() | ||
|  |      * @return string | ||
|  |      */ | ||
|  |     protected function _set_oembed_api($value) | ||
|  |     { | ||
|  |         if (!Validate::uri($value)) { | ||
|  |             throw new Services_oEmbed_Exception( | ||
|  |                 'API URI provided is invalid' | ||
|  |             ); | ||
|  |         } | ||
|  | 
 | ||
|  |         return $value; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Get the oEmbed response | ||
|  |      * | ||
|  |      * @param array $params Optional parameters for  | ||
|  |      *  | ||
|  |      * @throws {@link Services_oEmbed_Exception} on cURL errors | ||
|  |      * @throws {@link Services_oEmbed_Exception} on HTTP errors | ||
|  |      * @throws {@link Services_oEmbed_Exception} when result is not parsable  | ||
|  |      * @return object The oEmbed response as an object | ||
|  |      */ | ||
|  |     public function getObject(array $params = array()) | ||
|  |     { | ||
|  |         $params['url'] = $this->url->getURL(); | ||
|  |         if (!isset($params['format'])) { | ||
|  |             $params['format'] = 'json'; | ||
|  |         } | ||
|  | 
 | ||
|  |         $sets = array(); | ||
|  |         foreach ($params as $var => $val) { | ||
|  |             $sets[] = $var . '=' . urlencode($val); | ||
|  |         } | ||
|  | 
 | ||
|  |         $url = $this->options[self::OPTION_API] . '?' . implode('&', $sets); | ||
|  | 
 | ||
|  |         $ch = curl_init(); | ||
|  |         curl_setopt($ch, CURLOPT_URL, $url); | ||
|  |         curl_setopt($ch, CURLOPT_HEADER, false); | ||
|  |         curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); | ||
|  |         curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, $this->options[self::OPTION_TIMEOUT]); | ||
|  |         $result = curl_exec($ch); | ||
|  | 
 | ||
|  |         if (curl_errno($ch)) { | ||
|  |             throw new Services_oEmbed_Exception( | ||
|  |                 curl_error($ch), curl_errno($ch) | ||
|  |             ); | ||
|  |         } | ||
|  | 
 | ||
|  |         $code = curl_getinfo($ch, CURLINFO_HTTP_CODE); | ||
|  |         if (substr($code, 0, 1) != '2') { | ||
|  |             throw new Services_oEmbed_Exception('Non-200 code returned'); | ||
|  |         } | ||
|  | 
 | ||
|  |         curl_close($ch); | ||
|  | 
 | ||
|  |         switch ($params['format']) { | ||
|  |         case 'json': | ||
|  |             $res = json_decode($result); | ||
|  |             if (!is_object($res)) { | ||
|  |                 throw new Services_oEmbed_Exception( | ||
|  |                     'Could not parse JSON response' | ||
|  |                 ); | ||
|  |             } | ||
|  |             break; | ||
|  |         case 'xml': | ||
|  |             libxml_use_internal_errors(true); | ||
|  |             $res = simplexml_load_string($result); | ||
|  |             if (!$res instanceof SimpleXMLElement) { | ||
|  |                 $errors = libxml_get_errors(); | ||
|  |                 $err    = array_shift($errors); | ||
|  |                 libxml_clear_errors(); | ||
|  |                 libxml_use_internal_errors(false); | ||
|  |                 throw new Services_oEmbed_Exception( | ||
|  |                     $err->message, $error->code | ||
|  |                 ); | ||
|  |             } | ||
|  |             break; | ||
|  |         } | ||
|  | 
 | ||
|  |         return Services_oEmbed_Object::factory($res); | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Discover an oEmbed API  | ||
|  |      * | ||
|  |      * @param string $url The URL to attempt to discover oEmbed for | ||
|  |      * | ||
|  |      * @throws {@link Services_oEmbed_Exception} if the $url is invalid | ||
|  |      * @return string The oEmbed API endpoint discovered | ||
|  |      */ | ||
|  |     protected function discover($url) | ||
|  |     { | ||
|  |         $body = $this->sendRequest($url); | ||
|  | 
 | ||
|  |         // Find all <link /> tags that have a valid oembed type set. We then
 | ||
|  |         // extract the href attribute for each type.
 | ||
|  |         $regexp = '#<link([^>]*)type="' .  | ||
|  |                   '(application/json|text/xml)\+oembed"([^>]*)>#i'; | ||
|  | 
 | ||
|  |         $m = $ret = array(); | ||
|  |         if (!preg_match_all($regexp, $body, $m)) { | ||
|  |             throw new Services_oEmbed_Exception_NoSupport( | ||
|  |                 'No valid oEmbed links found on page' | ||
|  |             ); | ||
|  |         } | ||
|  | 
 | ||
|  |         foreach ($m[0] as $i => $link) { | ||
|  |             $h = array(); | ||
|  |             if (preg_match('/href="([^"]+)"/i', $link, $h)) { | ||
|  |                 $ret[$m[2][$i]] = $h[1]; | ||
|  |             } | ||
|  |         }  | ||
|  | 
 | ||
|  |         return (isset($ret['json']) ? $ret['json'] : array_pop($ret)); | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Send a GET request to the provider | ||
|  |      *  | ||
|  |      * @param mixed $url The URL to send the request to | ||
|  |      * | ||
|  |      * @throws {@link Services_oEmbed_Exception} on HTTP errors | ||
|  |      * @return string The contents of the response | ||
|  |      */ | ||
|  |     private function sendRequest($url) | ||
|  |     { | ||
|  |         $ch = curl_init(); | ||
|  |         curl_setopt($ch, CURLOPT_URL, $url); | ||
|  |         curl_setopt($ch, CURLOPT_HEADER, false); | ||
|  |         curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); | ||
|  |         curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, $this->options[self::OPTION_TIMEOUT]); | ||
|  |         curl_setopt($ch, CURLOPT_USERAGENT, $this->options[self::OPTION_USER_AGENT]); | ||
|  |         $result = curl_exec($ch); | ||
|  |         if (curl_errno($ch)) { | ||
|  |             throw new Services_oEmbed_Exception( | ||
|  |                 curl_error($ch), curl_errno($ch) | ||
|  |             ); | ||
|  |         } | ||
|  | 
 | ||
|  |         $code = curl_getinfo($ch, CURLINFO_HTTP_CODE); | ||
|  |         if (substr($code, 0, 1) != '2') { | ||
|  |             throw new Services_oEmbed_Exception('Non-200 code returned'); | ||
|  |         } | ||
|  | 
 | ||
|  |         return $result; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | ?>
 |