12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600 |
- <?php
- /**
- * This is the PHP Cloud Files API.
- *
- * <code>
- * # Authenticate to Cloud Files. The default is to automatically try
- * # to re-authenticate if an authentication token expires.
- * #
- * # NOTE: Some versions of cURL include an outdated certificate authority (CA)
- * # file. This API ships with a newer version obtained directly from
- * # cURL's web site (http://curl.haxx.se). To use the newer CA bundle,
- * # call the CF_Authentication instance's 'ssl_use_cabundle()' method.
- * #
- * $auth = new CF_Authentication($username, $api_key);
- * # $auth->ssl_use_cabundle(); # bypass cURL's old CA bundle
- * $auth->authenticate();
- *
- * # Establish a connection to the storage system
- * #
- * # NOTE: Some versions of cURL include an outdated certificate authority (CA)
- * # file. This API ships with a newer version obtained directly from
- * # cURL's web site (http://curl.haxx.se). To use the newer CA bundle,
- * # call the CF_Connection instance's 'ssl_use_cabundle()' method.
- * #
- * $conn = new CF_Connection($auth);
- * # $conn->ssl_use_cabundle(); # bypass cURL's old CA bundle
- *
- * # Create a remote Container and storage Object
- * #
- * $images = $conn->create_container("photos");
- * $bday = $images->create_object("first_birthday.jpg");
- *
- * # Upload content from a local file by streaming it. Note that we use
- * # a "float" for the file size to overcome PHP's 32-bit integer limit for
- * # very large files.
- * #
- * $fname = "/home/user/photos/birthdays/birthday1.jpg"; # filename to upload
- * $size = (float) sprintf("%u", filesize($fname));
- * $fp = open($fname, "r");
- * $bday->write($fp, $size);
- *
- * # Or... use a convenience function instead
- * #
- * $bday->load_from_filename("/home/user/photos/birthdays/birthday1.jpg");
- *
- * # Now, publish the "photos" container to serve the images by CDN.
- * # Use the "$uri" value to put in your web pages or send the link in an
- * # email message, etc.
- * #
- * $uri = $images->make_public();
- *
- * # Or... print out the Object's public URI
- * #
- * print $bday->public_uri();
- * </code>
- *
- * See the included tests directory for additional sample code.
- *
- * Requres PHP 5.x (for Exceptions and OO syntax) and PHP's cURL module.
- *
- * It uses the supporting "cloudfiles_http.php" module for HTTP(s) support and
- * allows for connection re-use and streaming of content into/out of Cloud Files
- * via PHP's cURL module.
- *
- * See COPYING for license information.
- *
- * @author Eric "EJ" Johnson <ej@racklabs.com>
- * @copyright Copyright (c) 2008, Rackspace US, Inc.
- * @package php-cloudfiles
- */
- /**
- */
- require_once("cloudfiles_exceptions.php");
- require("cloudfiles_http.php");
- define("DEFAULT_CF_API_VERSION", 1);
- define("MAX_CONTAINER_NAME_LEN", 256);
- define("MAX_OBJECT_NAME_LEN", 1024);
- define("MAX_OBJECT_SIZE", 5*1024*1024*1024+1);
- define("US_AUTHURL", "https://auth.api.rackspacecloud.com");
- define("UK_AUTHURL", "https://lon.auth.api.rackspacecloud.com");
- /**
- * Class for handling Cloud Files Authentication, call it's {@link authenticate()}
- * method to obtain authorized service urls and an authentication token.
- *
- * Example:
- * <code>
- * # Create the authentication instance
- * #
- * $auth = new CF_Authentication("username", "api_key");
- *
- * # NOTE: For UK Customers please specify your AuthURL Manually
- * # There is a Predfined constant to use EX:
- * #
- * # $auth = new CF_Authentication("username, "api_key", NULL, UK_AUTHURL);
- * # Using the UK_AUTHURL keyword will force the api to use the UK AuthUrl.
- * # rather then the US one. The NULL Is passed for legacy purposes and must
- * # be passed to function correctly.
- *
- * # NOTE: Some versions of cURL include an outdated certificate authority (CA)
- * # file. This API ships with a newer version obtained directly from
- * # cURL's web site (http://curl.haxx.se). To use the newer CA bundle,
- * # call the CF_Authentication instance's 'ssl_use_cabundle()' method.
- * #
- * # $auth->ssl_use_cabundle(); # bypass cURL's old CA bundle
- *
- * # Perform authentication request
- * #
- * $auth->authenticate();
- * </code>
- *
- * @package php-cloudfiles
- */
- class CF_Authentication
- {
- public $dbug;
- public $username;
- public $api_key;
- public $auth_host;
- public $account;
- /**
- * Instance variables that are set after successful authentication
- */
- public $storage_url;
- public $cdnm_url;
- public $auth_token;
- /**
- * Class constructor (PHP 5 syntax)
- *
- * @param string $username Mosso username
- * @param string $api_key Mosso API Access Key
- * @param string $account <i>Account name</i>
- * @param string $auth_host <i>Authentication service URI</i>
- */
- function __construct($username=NULL, $api_key=NULL, $account=NULL, $auth_host=US_AUTHURL)
- {
- $this->dbug = False;
- $this->username = $username;
- $this->api_key = $api_key;
- $this->account_name = $account;
- $this->auth_host = $auth_host;
- $this->storage_url = NULL;
- $this->cdnm_url = NULL;
- $this->auth_token = NULL;
- $this->cfs_http = new CF_Http(DEFAULT_CF_API_VERSION);
- }
- /**
- * Use the Certificate Authority bundle included with this API
- *
- * Most versions of PHP with cURL support include an outdated Certificate
- * Authority (CA) bundle (the file that lists all valid certificate
- * signing authorities). The SSL certificates used by the Cloud Files
- * storage system are perfectly valid but have been created/signed by
- * a CA not listed in these outdated cURL distributions.
- *
- * As a work-around, we've included an updated CA bundle obtained
- * directly from cURL's web site (http://curl.haxx.se). You can direct
- * the API to use this CA bundle by calling this method prior to making
- * any remote calls. The best place to use this method is right after
- * the CF_Authentication instance has been instantiated.
- *
- * You can specify your own CA bundle by passing in the full pathname
- * to the bundle. You can use the included CA bundle by leaving the
- * argument blank.
- *
- * @param string $path Specify path to CA bundle (default to included)
- */
- function ssl_use_cabundle($path=NULL)
- {
- $this->cfs_http->ssl_use_cabundle($path);
- }
- /**
- * Attempt to validate Username/API Access Key
- *
- * Attempts to validate credentials with the authentication service. It
- * either returns <kbd>True</kbd> or throws an Exception. Accepts a single
- * (optional) argument for the storage system API version.
- *
- * Example:
- * <code>
- * # Create the authentication instance
- * #
- * $auth = new CF_Authentication("username", "api_key");
- *
- * # Perform authentication request
- * #
- * $auth->authenticate();
- * </code>
- *
- * @param string $version API version for Auth service (optional)
- * @return boolean <kbd>True</kbd> if successfully authenticated
- * @throws AuthenticationException invalid credentials
- * @throws InvalidResponseException invalid response
- */
- function authenticate($version=DEFAULT_CF_API_VERSION)
- {
- list($status,$reason,$surl,$curl,$atoken) =
- $this->cfs_http->authenticate($this->username, $this->api_key,
- $this->account_name, $this->auth_host);
- if ($status == 401) {
- throw new AuthenticationException("Invalid username or access key.");
- }
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Unexpected response (".$status."): ".$reason);
- }
- if (!($surl || $curl) || !$atoken) {
- throw new InvalidResponseException(
- "Expected headers missing from auth service.");
- }
- $this->storage_url = $surl;
- $this->cdnm_url = $curl;
- $this->auth_token = $atoken;
- return True;
- }
- /**
- * Use Cached Token and Storage URL's rather then grabbing from the Auth System
- *
- * Example:
- * <code>
- * #Create an Auth instance
- * $auth = new CF_Authentication();
- * #Pass Cached URL's and Token as Args
- * $auth->load_cached_credentials("auth_token", "storage_url", "cdn_management_url");
- * </code>
- *
- * @param string $auth_token A Cloud Files Auth Token (Required)
- * @param string $storage_url The Cloud Files Storage URL (Required)
- * @param string $cdnm_url CDN Management URL (Required)
- * @return boolean <kbd>True</kbd> if successful
- * @throws SyntaxException If any of the Required Arguments are missing
- */
- function load_cached_credentials($auth_token, $storage_url, $cdnm_url)
- {
- if(!$storage_url || !$cdnm_url)
- {
- throw new SyntaxException("Missing Required Interface URL's!");
- return False;
- }
- if(!$auth_token)
- {
- throw new SyntaxException("Missing Auth Token!");
- return False;
- }
- $this->storage_url = $storage_url;
- $this->cdnm_url = $cdnm_url;
- $this->auth_token = $auth_token;
- return True;
- }
- /**
- * Grab Cloud Files info to be Cached for later use with the load_cached_credentials method.
- *
- * Example:
- * <code>
- * #Create an Auth instance
- * $auth = new CF_Authentication("UserName","API_Key");
- * $auth->authenticate();
- * $array = $auth->export_credentials();
- * </code>
- *
- * @return array of url's and an auth token.
- */
- function export_credentials()
- {
- $arr = array();
- $arr['storage_url'] = $this->storage_url;
- $arr['cdnm_url'] = $this->cdnm_url;
- $arr['auth_token'] = $this->auth_token;
- return $arr;
- }
- /**
- * Make sure the CF_Authentication instance has authenticated.
- *
- * Ensures that the instance variables necessary to communicate with
- * Cloud Files have been set from a previous authenticate() call.
- *
- * @return boolean <kbd>True</kbd> if successfully authenticated
- */
- function authenticated()
- {
- if (!($this->storage_url || $this->cdnm_url) || !$this->auth_token) {
- return False;
- }
- return True;
- }
- /**
- * Toggle debugging - set cURL verbose flag
- */
- function setDebug($bool)
- {
- $this->dbug = $bool;
- $this->cfs_http->setDebug($bool);
- }
- }
- /**
- * Class for establishing connections to the Cloud Files storage system.
- * Connection instances are used to communicate with the storage system at
- * the account level; listing and deleting Containers and returning Container
- * instances.
- *
- * Example:
- * <code>
- * # Create the authentication instance
- * #
- * $auth = new CF_Authentication("username", "api_key");
- *
- * # Perform authentication request
- * #
- * $auth->authenticate();
- *
- * # Create a connection to the storage/cdn system(s) and pass in the
- * # validated CF_Authentication instance.
- * #
- * $conn = new CF_Connection($auth);
- *
- * # NOTE: Some versions of cURL include an outdated certificate authority (CA)
- * # file. This API ships with a newer version obtained directly from
- * # cURL's web site (http://curl.haxx.se). To use the newer CA bundle,
- * # call the CF_Authentication instance's 'ssl_use_cabundle()' method.
- * #
- * # $conn->ssl_use_cabundle(); # bypass cURL's old CA bundle
- * </code>
- *
- * @package php-cloudfiles
- */
- class CF_Connection
- {
- public $dbug;
- public $cfs_http;
- public $cfs_auth;
- /**
- * Pass in a previously authenticated CF_Authentication instance.
- *
- * Example:
- * <code>
- * # Create the authentication instance
- * #
- * $auth = new CF_Authentication("username", "api_key");
- *
- * # Perform authentication request
- * #
- * $auth->authenticate();
- *
- * # Create a connection to the storage/cdn system(s) and pass in the
- * # validated CF_Authentication instance.
- * #
- * $conn = new CF_Connection($auth);
- *
- * # If you are connecting via Rackspace servers and have access
- * # to the servicenet network you can set the $servicenet to True
- * # like this.
- *
- * $conn = new CF_Connection($auth, $servicenet=True);
- *
- * </code>
- *
- * If the environement variable RACKSPACE_SERVICENET is defined it will
- * force to connect via the servicenet.
- *
- * @param obj $cfs_auth previously authenticated CF_Authentication instance
- * @param boolean $servicenet enable/disable access via Rackspace servicenet.
- * @throws AuthenticationException not authenticated
- */
- function __construct($cfs_auth, $servicenet=False)
- {
- if (isset($_ENV['RACKSPACE_SERVICENET']))
- $servicenet=True;
- $this->cfs_http = new CF_Http(DEFAULT_CF_API_VERSION);
- $this->cfs_auth = $cfs_auth;
- if (!$this->cfs_auth->authenticated()) {
- $e = "Need to pass in a previously authenticated ";
- $e .= "CF_Authentication instance.";
- throw new AuthenticationException($e);
- }
- $this->cfs_http->setCFAuth($this->cfs_auth, $servicenet=$servicenet);
- $this->dbug = False;
- }
- /**
- * Toggle debugging of instance and back-end HTTP module
- *
- * @param boolean $bool enable/disable cURL debugging
- */
- function setDebug($bool)
- {
- $this->dbug = (boolean) $bool;
- $this->cfs_http->setDebug($this->dbug);
- }
- /**
- * Close a connection
- *
- * Example:
- * <code>
- *
- * $conn->close();
- *
- * </code>
- *
- * Will close all current cUrl active connections.
- *
- */
- public function close()
- {
- $this->cfs_http->close();
- }
-
- /**
- * Cloud Files account information
- *
- * Return an array of two floats (since PHP only supports 32-bit integers);
- * number of containers on the account and total bytes used for the account.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * list($quantity, $bytes) = $conn->get_info();
- * print "Number of containers: " . $quantity . "\n";
- * print "Bytes stored in container: " . $bytes . "\n";
- * </code>
- *
- * @return array (number of containers, total bytes stored)
- * @throws InvalidResponseException unexpected response
- */
- function get_info()
- {
- list($status, $reason, $container_count, $total_bytes) =
- $this->cfs_http->head_account();
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->get_info();
- #}
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return array($container_count, $total_bytes);
- }
- /**
- * Create a Container
- *
- * Given a Container name, return a Container instance, creating a new
- * remote Container if it does not exit.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $images = $conn->create_container("my photos");
- * </code>
- *
- * @param string $container_name container name
- * @return CF_Container
- * @throws SyntaxException invalid name
- * @throws InvalidResponseException unexpected response
- */
- function create_container($container_name=NULL)
- {
- if ($container_name != "0" and !isset($container_name))
- throw new SyntaxException("Container name not set.");
-
- if (!isset($container_name) or $container_name == "")
- throw new SyntaxException("Container name not set.");
- if (strpos($container_name, "/") !== False) {
- $r = "Container name '".$container_name;
- $r .= "' cannot contain a '/' character.";
- throw new SyntaxException($r);
- }
- if (strlen($container_name) > MAX_CONTAINER_NAME_LEN) {
- throw new SyntaxException(sprintf(
- "Container name exeeds %d bytes.",
- MAX_CONTAINER_NAME_LEN));
- }
- $return_code = $this->cfs_http->create_container($container_name);
- if (!$return_code) {
- throw new InvalidResponseException("Invalid response ("
- . $return_code. "): " . $this->cfs_http->get_error());
- }
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->create_container($container_name);
- #}
- if ($return_code != 201 && $return_code != 202) {
- throw new InvalidResponseException(
- "Invalid response (".$return_code."): "
- . $this->cfs_http->get_error());
- }
- return new CF_Container($this->cfs_auth, $this->cfs_http, $container_name);
- }
- /**
- * Delete a Container
- *
- * Given either a Container instance or name, remove the remote Container.
- * The Container must be empty prior to removing it.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $conn->delete_container("my photos");
- * </code>
- *
- * @param string|obj $container container name or instance
- * @return boolean <kbd>True</kbd> if successfully deleted
- * @throws SyntaxException missing proper argument
- * @throws InvalidResponseException invalid response
- * @throws NonEmptyContainerException container not empty
- * @throws NoSuchContainerException remote container does not exist
- */
- function delete_container($container=NULL)
- {
- $container_name = NULL;
-
- if (is_object($container)) {
- if (get_class($container) == "CF_Container") {
- $container_name = $container->name;
- }
- }
- if (is_string($container)) {
- $container_name = $container;
- }
- if ($container_name != "0" and !isset($container_name))
- throw new SyntaxException("Must specify container object or name.");
- $return_code = $this->cfs_http->delete_container($container_name);
- if (!$return_code) {
- throw new InvalidResponseException("Failed to obtain http response");
- }
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->delete_container($container);
- #}
- if ($return_code == 409) {
- throw new NonEmptyContainerException(
- "Container must be empty prior to removing it.");
- }
- if ($return_code == 404) {
- throw new NoSuchContainerException(
- "Specified container did not exist to delete.");
- }
- if ($return_code != 204) {
- throw new InvalidResponseException(
- "Invalid response (".$return_code."): "
- . $this->cfs_http->get_error());
- }
- return True;
- }
- /**
- * Return a Container instance
- *
- * For the given name, return a Container instance if the remote Container
- * exists, otherwise throw a Not Found exception.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $images = $conn->get_container("my photos");
- * print "Number of Objects: " . $images->count . "\n";
- * print "Bytes stored in container: " . $images->bytes . "\n";
- * </code>
- *
- * @param string $container_name name of the remote Container
- * @return container CF_Container instance
- * @throws NoSuchContainerException thrown if no remote Container
- * @throws InvalidResponseException unexpected response
- */
- function get_container($container_name=NULL)
- {
- list($status, $reason, $count, $bytes) =
- $this->cfs_http->head_container($container_name);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->get_container($container_name);
- #}
- if ($status == 404) {
- throw new NoSuchContainerException("Container not found.");
- }
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response: ".$this->cfs_http->get_error());
- }
- return new CF_Container($this->cfs_auth, $this->cfs_http,
- $container_name, $count, $bytes);
- }
- /**
- * Return array of Container instances
- *
- * Return an array of CF_Container instances on the account. The instances
- * will be fully populated with Container attributes (bytes stored and
- * Object count)
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $clist = $conn->get_containers();
- * foreach ($clist as $cont) {
- * print "Container name: " . $cont->name . "\n";
- * print "Number of Objects: " . $cont->count . "\n";
- * print "Bytes stored in container: " . $cont->bytes . "\n";
- * }
- * </code>
- *
- * @return array An array of CF_Container instances
- * @throws InvalidResponseException unexpected response
- */
- function get_containers($limit=0, $marker=NULL)
- {
- list($status, $reason, $container_info) =
- $this->cfs_http->list_containers_info($limit, $marker);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->get_containers();
- #}
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response: ".$this->cfs_http->get_error());
- }
- $containers = array();
- foreach ($container_info as $name => $info) {
- $containers[] = new CF_Container($this->cfs_auth, $this->cfs_http,
- $info['name'], $info["count"], $info["bytes"], False);
- }
- return $containers;
- }
- /**
- * Return list of remote Containers
- *
- * Return an array of strings containing the names of all remote Containers.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $container_list = $conn->list_containers();
- * print_r($container_list);
- * Array
- * (
- * [0] => "my photos",
- * [1] => "my docs"
- * )
- * </code>
- *
- * @param integer $limit restrict results to $limit Containers
- * @param string $marker return results greater than $marker
- * @return array list of remote Containers
- * @throws InvalidResponseException unexpected response
- */
- function list_containers($limit=0, $marker=NULL)
- {
- list($status, $reason, $containers) =
- $this->cfs_http->list_containers($limit, $marker);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->list_containers($limit, $marker);
- #}
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return $containers;
- }
- /**
- * Return array of information about remote Containers
- *
- * Return a nested array structure of Container info.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- *
- * $container_info = $conn->list_containers_info();
- * print_r($container_info);
- * Array
- * (
- * ["my photos"] =>
- * Array
- * (
- * ["bytes"] => 78,
- * ["count"] => 2
- * )
- * ["docs"] =>
- * Array
- * (
- * ["bytes"] => 37323,
- * ["count"] => 12
- * )
- * )
- * </code>
- *
- * @param integer $limit restrict results to $limit Containers
- * @param string $marker return results greater than $marker
- * @return array nested array structure of Container info
- * @throws InvalidResponseException unexpected response
- */
- function list_containers_info($limit=0, $marker=NULL)
- {
- list($status, $reason, $container_info) =
- $this->cfs_http->list_containers_info($limit, $marker);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->list_containers_info($limit, $marker);
- #}
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return $container_info;
- }
- /**
- * Return list of Containers that have been published to the CDN.
- *
- * Return an array of strings containing the names of published Containers.
- * Note that this function returns the list of any Container that has
- * ever been CDN-enabled regardless of it's existence in the storage
- * system.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_containers = $conn->list_public_containers();
- * print_r($public_containers);
- * Array
- * (
- * [0] => "images",
- * [1] => "css",
- * [2] => "javascript"
- * )
- * </code>
- *
- * @param bool $enabled_only Will list all containers ever CDN enabled if * set to false or only currently enabled CDN containers if set to true. * Defaults to false.
- * @return array list of published Container names
- * @throws InvalidResponseException unexpected response
- */
- function list_public_containers($enabled_only=False)
- {
- list($status, $reason, $containers) =
- $this->cfs_http->list_cdn_containers($enabled_only);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->list_public_containers();
- #}
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return $containers;
- }
- /**
- * Set a user-supplied callback function to report download progress
- *
- * The callback function is used to report incremental progress of a data
- * download functions (e.g. $container->list_objects(), $obj->read(), etc).
- * The specified function will be periodically called with the number of
- * bytes transferred until the entire download is complete. This callback
- * function can be useful for implementing "progress bars" for large
- * downloads.
- *
- * The specified callback function should take a single integer parameter.
- *
- * <code>
- * function read_callback($bytes_transferred) {
- * print ">> downloaded " . $bytes_transferred . " bytes.\n";
- * # ... do other things ...
- * return;
- * }
- *
- * $conn = new CF_Connection($auth_obj);
- * $conn->set_read_progress_function("read_callback");
- * print_r($conn->list_containers());
- *
- * # output would look like this:
- * #
- * >> downloaded 10 bytes.
- * >> downloaded 11 bytes.
- * Array
- * (
- * [0] => fuzzy.txt
- * [1] => space name
- * )
- * </code>
- *
- * @param string $func_name the name of the user callback function
- */
- function set_read_progress_function($func_name)
- {
- $this->cfs_http->setReadProgressFunc($func_name);
- }
- /**
- * Set a user-supplied callback function to report upload progress
- *
- * The callback function is used to report incremental progress of a data
- * upload functions (e.g. $obj->write() call). The specified function will
- * be periodically called with the number of bytes transferred until the
- * entire upload is complete. This callback function can be useful
- * for implementing "progress bars" for large uploads/downloads.
- *
- * The specified callback function should take a single integer parameter.
- *
- * <code>
- * function write_callback($bytes_transferred) {
- * print ">> uploaded " . $bytes_transferred . " bytes.\n";
- * # ... do other things ...
- * return;
- * }
- *
- * $conn = new CF_Connection($auth_obj);
- * $conn->set_write_progress_function("write_callback");
- * $container = $conn->create_container("stuff");
- * $obj = $container->create_object("foo");
- * $obj->write("The callback function will be called during upload.");
- *
- * # output would look like this:
- * # >> uploaded 51 bytes.
- * #
- * </code>
- *
- * @param string $func_name the name of the user callback function
- */
- function set_write_progress_function($func_name)
- {
- $this->cfs_http->setWriteProgressFunc($func_name);
- }
- /**
- * Use the Certificate Authority bundle included with this API
- *
- * Most versions of PHP with cURL support include an outdated Certificate
- * Authority (CA) bundle (the file that lists all valid certificate
- * signing authorities). The SSL certificates used by the Cloud Files
- * storage system are perfectly valid but have been created/signed by
- * a CA not listed in these outdated cURL distributions.
- *
- * As a work-around, we've included an updated CA bundle obtained
- * directly from cURL's web site (http://curl.haxx.se). You can direct
- * the API to use this CA bundle by calling this method prior to making
- * any remote calls. The best place to use this method is right after
- * the CF_Authentication instance has been instantiated.
- *
- * You can specify your own CA bundle by passing in the full pathname
- * to the bundle. You can use the included CA bundle by leaving the
- * argument blank.
- *
- * @param string $path Specify path to CA bundle (default to included)
- */
- function ssl_use_cabundle($path=NULL)
- {
- $this->cfs_http->ssl_use_cabundle($path);
- }
- #private function _re_auth()
- #{
- # $new_auth = new CF_Authentication(
- # $this->cfs_auth->username,
- # $this->cfs_auth->api_key,
- # $this->cfs_auth->auth_host,
- # $this->cfs_auth->account);
- # $new_auth->authenticate();
- # $this->cfs_auth = $new_auth;
- # $this->cfs_http->setCFAuth($this->cfs_auth);
- # return True;
- #}
- }
- /**
- * Container operations
- *
- * Containers are storage compartments where you put your data (objects).
- * A container is similar to a directory or folder on a conventional filesystem
- * with the exception that they exist in a flat namespace, you can not create
- * containers inside of containers.
- *
- * You also have the option of marking a Container as "public" so that the
- * Objects stored in the Container are publicly available via the CDN.
- *
- * @package php-cloudfiles
- */
- class CF_Container
- {
- public $cfs_auth;
- public $cfs_http;
- public $name;
- public $object_count;
- public $bytes_used;
- public $cdn_enabled;
- public $cdn_streaming_uri;
- public $cdn_ssl_uri;
- public $cdn_uri;
- public $cdn_ttl;
- public $cdn_log_retention;
- public $cdn_acl_user_agent;
- public $cdn_acl_referrer;
- /**
- * Class constructor
- *
- * Constructor for Container
- *
- * @param obj $cfs_auth CF_Authentication instance
- * @param obj $cfs_http HTTP connection manager
- * @param string $name name of Container
- * @param int $count number of Objects stored in this Container
- * @param int $bytes number of bytes stored in this Container
- * @throws SyntaxException invalid Container name
- */
- function __construct(&$cfs_auth, &$cfs_http, $name, $count=0,
- $bytes=0, $docdn=True)
- {
- if (strlen($name) > MAX_CONTAINER_NAME_LEN) {
- throw new SyntaxException("Container name exceeds "
- . "maximum allowed length.");
- }
- if (strpos($name, "/") !== False) {
- throw new SyntaxException(
- "Container names cannot contain a '/' character.");
- }
- $this->cfs_auth = $cfs_auth;
- $this->cfs_http = $cfs_http;
- $this->name = $name;
- $this->object_count = $count;
- $this->bytes_used = $bytes;
- $this->cdn_enabled = NULL;
- $this->cdn_uri = NULL;
- $this->cdn_ssl_uri = NULL;
- $this->cdn_streaming_uri = NULL;
- $this->cdn_ttl = NULL;
- $this->cdn_log_retention = NULL;
- $this->cdn_acl_user_agent = NULL;
- $this->cdn_acl_referrer = NULL;
- if ($this->cfs_http->getCDNMUrl() != NULL && $docdn) {
- $this->_cdn_initialize();
- }
- }
- /**
- * String representation of Container
- *
- * Pretty print the Container instance.
- *
- * @return string Container details
- */
- function __toString()
- {
- $me = sprintf("name: %s, count: %.0f, bytes: %.0f",
- $this->name, $this->object_count, $this->bytes_used);
- if ($this->cfs_http->getCDNMUrl() != NULL) {
- $me .= sprintf(", cdn: %s, cdn uri: %s, cdn ttl: %.0f, logs retention: %s",
- $this->is_public() ? "Yes" : "No",
- $this->cdn_uri, $this->cdn_ttl,
- $this->cdn_log_retention ? "Yes" : "No"
- );
- if ($this->cdn_acl_user_agent != NULL) {
- $me .= ", cdn acl user agent: " . $this->cdn_acl_user_agent;
- }
- if ($this->cdn_acl_referrer != NULL) {
- $me .= ", cdn acl referrer: " . $this->cdn_acl_referrer;
- }
-
-
- }
- return $me;
- }
- /**
- * Enable Container content to be served via CDN or modify CDN attributes
- *
- * Either enable this Container's content to be served via CDN or
- * adjust its CDN attributes. This Container will always return the
- * same CDN-enabled URI each time it is toggled public/private/public.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->create_container("public");
- *
- * # CDN-enable the container and set it's TTL for a month
- * #
- * $public_container->make_public(86400/2); # 12 hours (86400 seconds/day)
- * </code>
- *
- * @param int $ttl the time in seconds content will be cached in the CDN
- * @returns string the CDN enabled Container's URI
- * @throws CDNNotEnabledException CDN functionality not returned during auth
- * @throws AuthenticationException if auth token is not valid/expired
- * @throws InvalidResponseException unexpected response
- */
- function make_public($ttl=86400)
- {
- if ($this->cfs_http->getCDNMUrl() == NULL) {
- throw new CDNNotEnabledException(
- "Authentication response did not indicate CDN availability");
- }
- if ($this->cdn_uri != NULL) {
- # previously published, assume we're setting new attributes
- list($status, $reason, $cdn_uri, $cdn_ssl_uri) =
- $this->cfs_http->update_cdn_container($this->name,$ttl,
- $this->cdn_log_retention,
- $this->cdn_acl_user_agent,
- $this->cdn_acl_referrer);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->make_public($ttl);
- #}
- if ($status == 404) {
- # this instance _thinks_ the container was published, but the
- # cdn management system thinks otherwise - try again with a PUT
- list($status, $reason, $cdn_uri, $cdn_ssl_uri) =
- $this->cfs_http->add_cdn_container($this->name,$ttl);
- }
- } else {
- # publish it for first time
- list($status, $reason, $cdn_uri, $cdn_ssl_uri) =
- $this->cfs_http->add_cdn_container($this->name,$ttl);
- }
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->make_public($ttl);
- #}
- if (!in_array($status, array(201,202))) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- $this->cdn_enabled = True;
- $this->cdn_ttl = $ttl;
- $this->cdn_ssl_uri = $cdn_ssl_uri;
- $this->cdn_uri = $cdn_uri;
- $this->cdn_log_retention = False;
- $this->cdn_acl_user_agent = "";
- $this->cdn_acl_referrer = "";
- return $this->cdn_uri;
- }
- /**
- * Purge Containers objects from CDN Cache.
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- * $container = $conn->get_container("cdn_enabled");
- * $container->purge_from_cdn("user@domain.com");
- * # or
- * $container->purge_from_cdn();
- * # or
- * $container->purge_from_cdn("user1@domain.com,user2@domain.com");
- * @returns boolean True if successful
- * @throws CDNNotEnabledException if CDN Is not enabled on this connection
- * @throws InvalidResponseException if the response expected is not returned
- */
- function purge_from_cdn($email=null)
- {
- if (!$this->cfs_http->getCDNMUrl())
- {
- throw new CDNNotEnabledException(
- "Authentication response did not indicate CDN availability");
- }
- $status = $this->cfs_http->purge_from_cdn($this->name, $email);
- if ($status < 199 or $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return True;
- }
- /**
- * Enable ACL restriction by User Agent for this container.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->get_container("public");
- *
- * # Enable ACL by Referrer
- * $public_container->acl_referrer("Mozilla");
- * </code>
- *
- * @returns boolean True if successful
- * @throws CDNNotEnabledException CDN functionality not returned during auth
- * @throws AuthenticationException if auth token is not valid/expired
- * @throws InvalidResponseException unexpected response
- */
- function acl_user_agent($cdn_acl_user_agent="") {
- if ($this->cfs_http->getCDNMUrl() == NULL) {
- throw new CDNNotEnabledException(
- "Authentication response did not indicate CDN availability");
- }
- list($status,$reason) =
- $this->cfs_http->update_cdn_container($this->name,
- $this->cdn_ttl,
- $this->cdn_log_retention,
- $cdn_acl_user_agent,
- $this->cdn_acl_referrer
- );
- if (!in_array($status, array(202,404))) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- $this->cdn_acl_user_agent = $cdn_acl_user_agent;
- return True;
- }
- /**
- * Enable ACL restriction by referer for this container.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->get_container("public");
- *
- * # Enable Referrer
- * $public_container->acl_referrer("http://www.example.com/gallery.php");
- * </code>
- *
- * @returns boolean True if successful
- * @throws CDNNotEnabledException CDN functionality not returned during auth
- * @throws AuthenticationException if auth token is not valid/expired
- * @throws InvalidResponseException unexpected response
- */
- function acl_referrer($cdn_acl_referrer="") {
- if ($this->cfs_http->getCDNMUrl() == NULL) {
- throw new CDNNotEnabledException(
- "Authentication response did not indicate CDN availability");
- }
- list($status,$reason) =
- $this->cfs_http->update_cdn_container($this->name,
- $this->cdn_ttl,
- $this->cdn_log_retention,
- $this->cdn_acl_user_agent,
- $cdn_acl_referrer
- );
- if (!in_array($status, array(202,404))) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- $this->cdn_acl_referrer = $cdn_acl_referrer;
- return True;
- }
-
- /**
- * Enable log retention for this CDN container.
- *
- * Enable CDN log retention on the container. If enabled logs will
- * be periodically (at unpredictable intervals) compressed and
- * uploaded to a ".CDN_ACCESS_LOGS" container in the form of
- * "container_name.YYYYMMDDHH-XXXX.gz". Requires CDN be enabled on
- * the account.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->get_container("public");
- *
- * # Enable logs retention.
- * $public_container->log_retention(True);
- * </code>
- *
- * @returns boolean True if successful
- * @throws CDNNotEnabledException CDN functionality not returned during auth
- * @throws AuthenticationException if auth token is not valid/expired
- * @throws InvalidResponseException unexpected response
- */
- function log_retention($cdn_log_retention=False) {
- if ($this->cfs_http->getCDNMUrl() == NULL) {
- throw new CDNNotEnabledException(
- "Authentication response did not indicate CDN availability");
- }
- list($status,$reason) =
- $this->cfs_http->update_cdn_container($this->name,
- $this->cdn_ttl,
- $cdn_log_retention,
- $this->cdn_acl_user_agent,
- $this->cdn_acl_referrer
- );
- if (!in_array($status, array(202,404))) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- $this->cdn_log_retention = $cdn_log_retention;
- return True;
- }
-
- /**
- * Disable the CDN sharing for this container
- *
- * Use this method to disallow distribution into the CDN of this Container's
- * content.
- *
- * NOTE: Any content already cached in the CDN will continue to be served
- * from its cache until the TTL expiration transpires. The default
- * TTL is typically one day, so "privatizing" the Container will take
- * up to 24 hours before the content is purged from the CDN cache.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->get_container("public");
- *
- * # Disable CDN accessability
- * # ... still cached up to a month based on previous example
- * #
- * $public_container->make_private();
- * </code>
- *
- * @returns boolean True if successful
- * @throws CDNNotEnabledException CDN functionality not returned during auth
- * @throws AuthenticationException if auth token is not valid/expired
- * @throws InvalidResponseException unexpected response
- */
- function make_private()
- {
- if ($this->cfs_http->getCDNMUrl() == NULL) {
- throw new CDNNotEnabledException(
- "Authentication response did not indicate CDN availability");
- }
- list($status,$reason) = $this->cfs_http->remove_cdn_container($this->name);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->make_private();
- #}
- if (!in_array($status, array(202,404))) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- $this->cdn_enabled = False;
- $this->cdn_ttl = NULL;
- $this->cdn_uri = NULL;
- $this->cdn_ssl_uri = NULL;
- $this->cdn_streaming_uri - NULL;
- $this->cdn_log_retention = NULL;
- $this->cdn_acl_user_agent = NULL;
- $this->cdn_acl_referrer = NULL;
- return True;
- }
- /**
- * Check if this Container is being publicly served via CDN
- *
- * Use this method to determine if the Container's content is currently
- * available through the CDN.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->get_container("public");
- *
- * # Display CDN accessability
- * #
- * $public_container->is_public() ? print "Yes" : print "No";
- * </code>
- *
- * @returns boolean True if enabled, False otherwise
- */
- function is_public()
- {
- return $this->cdn_enabled == True ? True : False;
- }
- /**
- * Create a new remote storage Object
- *
- * Return a new Object instance. If the remote storage Object exists,
- * the instance's attributes are populated.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->get_container("public");
- *
- * # This creates a local instance of a storage object but only creates
- * # it in the storage system when the object's write() method is called.
- * #
- * $pic = $public_container->create_object("baby.jpg");
- * </code>
- *
- * @param string $obj_name name of storage Object
- * @return obj CF_Object instance
- */
- function create_object($obj_name=NULL)
- {
- return new CF_Object($this, $obj_name);
- }
- /**
- * Return an Object instance for the remote storage Object
- *
- * Given a name, return a Object instance representing the
- * remote storage object.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $public_container = $conn->get_container("public");
- *
- * # This call only fetches header information and not the content of
- * # the storage object. Use the Object's read() or stream() methods
- * # to obtain the object's data.
- * #
- * $pic = $public_container->get_object("baby.jpg");
- * </code>
- *
- * @param string $obj_name name of storage Object
- * @return obj CF_Object instance
- */
- function get_object($obj_name=NULL)
- {
- return new CF_Object($this, $obj_name, True);
- }
- /**
- * Return a list of Objects
- *
- * Return an array of strings listing the Object names in this Container.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $images = $conn->get_container("my photos");
- *
- * # Grab the list of all storage objects
- * #
- * $all_objects = $images->list_objects();
- *
- * # Grab subsets of all storage objects
- * #
- * $first_ten = $images->list_objects(10);
- *
- * # Note the use of the previous result's last object name being
- * # used as the 'marker' parameter to fetch the next 10 objects
- * #
- * $next_ten = $images->list_objects(10, $first_ten[count($first_ten)-1]);
- *
- * # Grab images starting with "birthday_party" and default limit/marker
- * # to match all photos with that prefix
- * #
- * $prefixed = $images->list_objects(0, NULL, "birthday");
- *
- * # Assuming you have created the appropriate directory marker Objects,
- * # you can traverse your pseudo-hierarchical containers
- * # with the "path" argument.
- * #
- * $animals = $images->list_objects(0,NULL,NULL,"pictures/animals");
- * $dogs = $images->list_objects(0,NULL,NULL,"pictures/animals/dogs");
- * </code>
- *
- * @param int $limit <i>optional</i> only return $limit names
- * @param int $marker <i>optional</i> subset of names starting at $marker
- * @param string $prefix <i>optional</i> Objects whose names begin with $prefix
- * @param string $path <i>optional</i> only return results under "pathname"
- * @return array array of strings
- * @throws InvalidResponseException unexpected response
- */
- function list_objects($limit=0, $marker=NULL, $prefix=NULL, $path=NULL)
- {
- list($status, $reason, $obj_list) =
- $this->cfs_http->list_objects($this->name, $limit,
- $marker, $prefix, $path);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->list_objects($limit, $marker, $prefix, $path);
- #}
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return $obj_list;
- }
- /**
- * Return an array of Objects
- *
- * Return an array of Object instances in this Container.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $images = $conn->get_container("my photos");
- *
- * # Grab the list of all storage objects
- * #
- * $all_objects = $images->get_objects();
- *
- * # Grab subsets of all storage objects
- * #
- * $first_ten = $images->get_objects(10);
- *
- * # Note the use of the previous result's last object name being
- * # used as the 'marker' parameter to fetch the next 10 objects
- * #
- * $next_ten = $images->list_objects(10, $first_ten[count($first_ten)-1]);
- *
- * # Grab images starting with "birthday_party" and default limit/marker
- * # to match all photos with that prefix
- * #
- * $prefixed = $images->get_objects(0, NULL, "birthday");
- *
- * # Assuming you have created the appropriate directory marker Objects,
- * # you can traverse your pseudo-hierarchical containers
- * # with the "path" argument.
- * #
- * $animals = $images->get_objects(0,NULL,NULL,"pictures/animals");
- * $dogs = $images->get_objects(0,NULL,NULL,"pictures/animals/dogs");
- * </code>
- *
- * @param int $limit <i>optional</i> only return $limit names
- * @param int $marker <i>optional</i> subset of names starting at $marker
- * @param string $prefix <i>optional</i> Objects whose names begin with $prefix
- * @param string $path <i>optional</i> only return results under "pathname"
- * @return array array of strings
- * @throws InvalidResponseException unexpected response
- */
- function get_objects($limit=0, $marker=NULL, $prefix=NULL, $path=NULL)
- {
- list($status, $reason, $obj_array) =
- $this->cfs_http->get_objects($this->name, $limit,
- $marker, $prefix, $path);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->get_objects($limit, $marker, $prefix, $path);
- #}
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- $objects = array();
- foreach ($obj_array as $obj) {
- $tmp = new CF_Object($this, $obj["name"], False, False);
- $tmp->content_type = $obj["content_type"];
- $tmp->content_length = (float) $obj["bytes"];
- $tmp->set_etag($obj["hash"]);
- $tmp->last_modified = $obj["last_modified"];
- $objects[] = $tmp;
- }
- return $objects;
- }
- /**
- * Copy a remote storage Object to a target Container
- *
- * Given an Object instance or name and a target Container instance or name, copy copies the remote Object
- * and all associated metadata.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $images = $conn->get_container("my photos");
- *
- * # Copy specific object
- * #
- * $images->copy_object_to("disco_dancing.jpg","container_target");
- * </code>
- *
- * @param obj $obj name or instance of Object to copy
- * @param obj $container_target name or instance of target Container
- * @param string $dest_obj_name name of target object (optional - uses source name if omitted)
- * @param array $metadata metadata array for new object (optional)
- * @param array $headers header fields array for the new object (optional)
- * @return boolean <kbd>true</kbd> if successfully copied
- * @throws SyntaxException invalid Object/Container name
- * @throws NoSuchObjectException remote Object does not exist
- * @throws InvalidResponseException unexpected response
- */
- function copy_object_to($obj,$container_target,$dest_obj_name=NULL,$metadata=NULL,$headers=NULL)
- {
- $obj_name = NULL;
- if (is_object($obj)) {
- if (get_class($obj) == "CF_Object") {
- $obj_name = $obj->name;
- }
- }
- if (is_string($obj)) {
- $obj_name = $obj;
- }
- if (!$obj_name) {
- throw new SyntaxException("Object name not set.");
- }
- if ($dest_obj_name === NULL) {
- $dest_obj_name = $obj_name;
- }
- $container_name_target = NULL;
- if (is_object($container_target)) {
- if (get_class($container_target) == "CF_Container") {
- $container_name_target = $container_target->name;
- }
- }
- if (is_string($container_target)) {
- $container_name_target = $container_target;
- }
- if (!$container_name_target) {
- throw new SyntaxException("Container name target not set.");
- }
- $status = $this->cfs_http->copy_object($obj_name,$dest_obj_name,$this->name,$container_name_target,$metadata,$headers);
- if ($status == 404) {
- $m = "Specified object '".$this->name."/".$obj_name;
- $m.= "' did not exist as source to copy from or '".$container_name_target."' did not exist as target to copy to.";
- throw new NoSuchObjectException($m);
- }
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return true;
- }
- /**
- * Copy a remote storage Object from a source Container
- *
- * Given an Object instance or name and a source Container instance or name, copy copies the remote Object
- * and all associated metadata.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $images = $conn->get_container("my photos");
- *
- * # Copy specific object
- * #
- * $images->copy_object_from("disco_dancing.jpg","container_source");
- * </code>
- *
- * @param obj $obj name or instance of Object to copy
- * @param obj $container_source name or instance of source Container
- * @param string $dest_obj_name name of target object (optional - uses source name if omitted)
- * @param array $metadata metadata array for new object (optional)
- * @param array $headers header fields array for the new object (optional)
- * @return boolean <kbd>true</kbd> if successfully copied
- * @throws SyntaxException invalid Object/Container name
- * @throws NoSuchObjectException remote Object does not exist
- * @throws InvalidResponseException unexpected response
- */
- function copy_object_from($obj,$container_source,$dest_obj_name=NULL,$metadata=NULL,$headers=NULL)
- {
- $obj_name = NULL;
- if (is_object($obj)) {
- if (get_class($obj) == "CF_Object") {
- $obj_name = $obj->name;
- }
- }
- if (is_string($obj)) {
- $obj_name = $obj;
- }
- if (!$obj_name) {
- throw new SyntaxException("Object name not set.");
- }
- if ($dest_obj_name === NULL) {
- $dest_obj_name = $obj_name;
- }
- $container_name_source = NULL;
- if (is_object($container_source)) {
- if (get_class($container_source) == "CF_Container") {
- $container_name_source = $container_source->name;
- }
- }
- if (is_string($container_source)) {
- $container_name_source = $container_source;
- }
- if (!$container_name_source) {
- throw new SyntaxException("Container name source not set.");
- }
- $status = $this->cfs_http->copy_object($obj_name,$dest_obj_name,$container_name_source,$this->name,$metadata,$headers);
- if ($status == 404) {
- $m = "Specified object '".$container_name_source."/".$obj_name;
- $m.= "' did not exist as source to copy from or '".$this->name."/".$obj_name."' did not exist as target to copy to.";
- throw new NoSuchObjectException($m);
- }
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
-
- return true;
- }
- /**
- * Move a remote storage Object to a target Container
- *
- * Given an Object instance or name and a target Container instance or name, move copies the remote Object
- * and all associated metadata and deletes the source Object afterwards
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $images = $conn->get_container("my photos");
- *
- * # Move specific object
- * #
- * $images->move_object_to("disco_dancing.jpg","container_target");
- * </code>
- *
- * @param obj $obj name or instance of Object to move
- * @param obj $container_target name or instance of target Container
- * @param string $dest_obj_name name of target object (optional - uses source name if omitted)
- * @param array $metadata metadata array for new object (optional)
- * @param array $headers header fields array for the new object (optional)
- * @return boolean <kbd>true</kbd> if successfully moved
- * @throws SyntaxException invalid Object/Container name
- * @throws NoSuchObjectException remote Object does not exist
- * @throws InvalidResponseException unexpected response
- */
- function move_object_to($obj,$container_target,$dest_obj_name=NULL,$metadata=NULL,$headers=NULL)
- {
- $retVal = false;
- if(self::copy_object_to($obj,$container_target,$dest_obj_name,$metadata,$headers)) {
- $retVal = self::delete_object($obj,$this->name);
- }
- return $retVal;
- }
- /**
- * Move a remote storage Object from a source Container
- *
- * Given an Object instance or name and a source Container instance or name, move copies the remote Object
- * and all associated metadata and deletes the source Object afterwards
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $images = $conn->get_container("my photos");
- *
- * # Move specific object
- * #
- * $images->move_object_from("disco_dancing.jpg","container_target");
- * </code>
- *
- * @param obj $obj name or instance of Object to move
- * @param obj $container_source name or instance of target Container
- * @param string $dest_obj_name name of target object (optional - uses source name if omitted)
- * @param array $metadata metadata array for new object (optional)
- * @param array $headers header fields array for the new object (optional)
- * @return boolean <kbd>true</kbd> if successfully moved
- * @throws SyntaxException invalid Object/Container name
- * @throws NoSuchObjectException remote Object does not exist
- * @throws InvalidResponseException unexpected response
- */
- function move_object_from($obj,$container_source,$dest_obj_name=NULL,$metadata=NULL,$headers=NULL)
- {
- $retVal = false;
- if(self::copy_object_from($obj,$container_source,$dest_obj_name,$metadata,$headers)) {
- $retVal = self::delete_object($obj,$container_source);
- }
- return $retVal;
- }
- /**
- * Delete a remote storage Object
- *
- * Given an Object instance or name, permanently remove the remote Object
- * and all associated metadata.
- *
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- *
- * $images = $conn->get_container("my photos");
- *
- * # Delete specific object
- * #
- * $images->delete_object("disco_dancing.jpg");
- * </code>
- *
- * @param obj $obj name or instance of Object to delete
- * @param obj $container name or instance of Container in which the object resides (optional)
- * @return boolean <kbd>True</kbd> if successfully removed
- * @throws SyntaxException invalid Object name
- * @throws NoSuchObjectException remote Object does not exist
- * @throws InvalidResponseException unexpected response
- */
- function delete_object($obj,$container=NULL)
- {
- $obj_name = NULL;
- if (is_object($obj)) {
- if (get_class($obj) == "CF_Object") {
- $obj_name = $obj->name;
- }
- }
- if (is_string($obj)) {
- $obj_name = $obj;
- }
- if (!$obj_name) {
- throw new SyntaxException("Object name not set.");
- }
- $container_name = NULL;
- if($container === NULL) {
- $container_name = $this->name;
- }
- else {
- if (is_object($container)) {
- if (get_class($container) == "CF_Container") {
- $container_name = $container->name;
- }
- }
- if (is_string($container)) {
- $container_name = $container;
- }
- if (!$container_name) {
- throw new SyntaxException("Container name source not set.");
- }
- }
- $status = $this->cfs_http->delete_object($container_name, $obj_name);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->delete_object($obj);
- #}
- if ($status == 404) {
- $m = "Specified object '".$container_name."/".$obj_name;
- $m.= "' did not exist to delete.";
- throw new NoSuchObjectException($m);
- }
- if ($status != 204) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- return True;
- }
- /**
- * Helper function to create "path" elements for a given Object name
- *
- * Given an Object whos name contains '/' path separators, this function
- * will create the "directory marker" Objects of one byte with the
- * Content-Type of "application/directory".
- *
- * It assumes the last element of the full path is the "real" Object
- * and does NOT create a remote storage Object for that last element.
- */
- function create_paths($path_name)
- {
- if ($path_name[0] == '/') {
- $path_name = mb_substr($path_name, 0, 1);
- }
- $elements = explode('/', $path_name, -1);
- $build_path = "";
- foreach ($elements as $idx => $val) {
- if (!$build_path) {
- $build_path = $val;
- } else {
- $build_path .= "/" . $val;
- }
- $obj = new CF_Object($this, $build_path);
- $obj->content_type = "application/directory";
- $obj->write(".", 1);
- }
- }
- /**
- * Internal method to grab CDN/Container info if appropriate to do so
- *
- * @throws InvalidResponseException unexpected response
- */
- private function _cdn_initialize()
- {
- list($status, $reason, $cdn_enabled, $cdn_ssl_uri, $cdn_streaming_uri, $cdn_uri, $cdn_ttl,
- $cdn_log_retention, $cdn_acl_user_agent, $cdn_acl_referrer) =
- $this->cfs_http->head_cdn_container($this->name);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->_cdn_initialize();
- #}
- if (!in_array($status, array(204,404))) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->cfs_http->get_error());
- }
- $this->cdn_enabled = $cdn_enabled;
- $this->cdn_streaming_uri = $cdn_streaming_uri;
- $this->cdn_ssl_uri = $cdn_ssl_uri;
- $this->cdn_uri = $cdn_uri;
- $this->cdn_ttl = $cdn_ttl;
- $this->cdn_log_retention = $cdn_log_retention;
- $this->cdn_acl_user_agent = $cdn_acl_user_agent;
- $this->cdn_acl_referrer = $cdn_acl_referrer;
- }
- #private function _re_auth()
- #{
- # $new_auth = new CF_Authentication(
- # $this->cfs_auth->username,
- # $this->cfs_auth->api_key,
- # $this->cfs_auth->auth_host,
- # $this->cfs_auth->account);
- # $new_auth->authenticate();
- # $this->cfs_auth = $new_auth;
- # $this->cfs_http->setCFAuth($this->cfs_auth);
- # return True;
- #}
- }
- /**
- * Object operations
- *
- * An Object is analogous to a file on a conventional filesystem. You can
- * read data from, or write data to your Objects. You can also associate
- * arbitrary metadata with them.
- *
- * @package php-cloudfiles
- */
- class CF_Object
- {
- public $container;
- public $name;
- public $last_modified;
- public $content_type;
- public $content_length;
- public $metadata;
- public $headers;
- public $manifest;
- private $etag;
- /**
- * Class constructor
- *
- * @param obj $container CF_Container instance
- * @param string $name name of Object
- * @param boolean $force_exists if set, throw an error if Object doesn't exist
- */
- function __construct(&$container, $name, $force_exists=False, $dohead=True)
- {
- if ($name[0] == "/") {
- $r = "Object name '".$name;
- $r .= "' cannot contain begin with a '/' character.";
- throw new SyntaxException($r);
- }
- if (strlen($name) > MAX_OBJECT_NAME_LEN) {
- throw new SyntaxException("Object name exceeds "
- . "maximum allowed length.");
- }
- $this->container = $container;
- $this->name = $name;
- $this->etag = NULL;
- $this->_etag_override = False;
- $this->last_modified = NULL;
- $this->content_type = NULL;
- $this->content_length = 0;
- $this->metadata = array();
- $this->headers = array();
- $this->manifest = NULL;
- if ($dohead) {
- if (!$this->_initialize() && $force_exists) {
- throw new NoSuchObjectException("No such object '".$name."'");
- }
- }
- }
- /**
- * String representation of Object
- *
- * Pretty print the Object's location and name
- *
- * @return string Object information
- */
- function __toString()
- {
- return $this->container->name . "/" . $this->name;
- }
- /**
- * Internal check to get the proper mimetype.
- *
- * This function would go over the available PHP methods to get
- * the MIME type.
- *
- * By default it will try to use the PHP fileinfo library which is
- * available from PHP 5.3 or as an PECL extension
- * (http://pecl.php.net/package/Fileinfo).
- *
- * It will get the magic file by default from the system wide file
- * which is usually available in /usr/share/magic on Unix or try
- * to use the file specified in the source directory of the API
- * (share directory).
- *
- * if fileinfo is not available it will try to use the internal
- * mime_content_type function.
- *
- * @param string $handle name of file or buffer to guess the type from
- * @return boolean <kbd>True</kbd> if successful
- * @throws BadContentTypeException
- */
- function _guess_content_type($handle) {
- if ($this->content_type)
- return;
-
- // if (function_exists("finfo_open")) {
- // $local_magic = dirname(__FILE__) . "/share/magic";
- // $finfo = @finfo_open(FILEINFO_MIME, $local_magic);
- //
- // if (!$finfo)
- // $finfo = @finfo_open(FILEINFO_MIME);
- //
- // if ($finfo) {
- //
- // if (is_file((string)$handle))
- // $ct = @finfo_file($finfo, $handle);
- // else
- // $ct = @finfo_buffer($finfo, $handle);
- //
- // /* PHP 5.3 fileinfo display extra information like
- // charset so we remove everything after the ; since
- // we are not into that stuff */
- // if ($ct) {
- // $extra_content_type_info = strpos($ct, "; ");
- // if ($extra_content_type_info)
- // $ct = substr($ct, 0, $extra_content_type_info);
- // }
- //
- // if ($ct && $ct != 'application/octet-stream')
- // $this->content_type = $ct;
- //
- // @finfo_close($finfo);
- // }
- // }
- //
- // if (!$this->content_type && (string)is_file($handle) && function_exists("mime_content_type")) {
- // $this->content_type = @mime_content_type($handle);
- // }
- //use OC's mimetype detection for files
- if(is_file($handle)){
- $this->content_type=OC_Helper::getMimeType($handle);
- }else{
- $this->content_type=OC_Helper::getStringMimeType($handle);
- }
- if (!$this->content_type) {
- throw new BadContentTypeException("Required Content-Type not set");
- }
- return True;
- }
-
- /**
- * String representation of the Object's public URI
- *
- * A string representing the Object's public URI assuming that it's
- * parent Container is CDN-enabled.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * # Print out the Object's CDN URI (if it has one) in an HTML img-tag
- * #
- * print "<img src='$pic->public_uri()' />\n";
- * </code>
- *
- * @return string Object's public URI or NULL
- */
- function public_uri()
- {
- if ($this->container->cdn_enabled) {
- return $this->container->cdn_uri . "/" . $this->name;
- }
- return NULL;
- }
- /**
- * String representation of the Object's public SSL URI
- *
- * A string representing the Object's public SSL URI assuming that it's
- * parent Container is CDN-enabled.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * # Print out the Object's CDN SSL URI (if it has one) in an HTML img-tag
- * #
- * print "<img src='$pic->public_ssl_uri()' />\n";
- * </code>
- *
- * @return string Object's public SSL URI or NULL
- */
- function public_ssl_uri()
- {
- if ($this->container->cdn_enabled) {
- return $this->container->cdn_ssl_uri . "/" . $this->name;
- }
- return NULL;
- }
- /**
- * String representation of the Object's public Streaming URI
- *
- * A string representing the Object's public Streaming URI assuming that it's
- * parent Container is CDN-enabled.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * # Print out the Object's CDN Streaming URI (if it has one) in an HTML img-tag
- * #
- * print "<img src='$pic->public_streaming_uri()' />\n";
- * </code>
- *
- * @return string Object's public Streaming URI or NULL
- */
- function public_streaming_uri()
- {
- if ($this->container->cdn_enabled) {
- return $this->container->cdn_streaming_uri . "/" . $this->name;
- }
- return NULL;
- }
- /**
- * Read the remote Object's data
- *
- * Returns the Object's data. This is useful for smaller Objects such
- * as images or office documents. Object's with larger content should use
- * the stream() method below.
- *
- * Pass in $hdrs array to set specific custom HTTP headers such as
- * If-Match, If-None-Match, If-Modified-Since, Range, etc.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * $my_docs = $conn->get_container("documents");
- * $doc = $my_docs->get_object("README");
- * $data = $doc->read(); # read image content into a string variable
- * print $data;
- *
- * # Or see stream() below for a different example.
- * #
- * </code>
- *
- * @param array $hdrs user-defined headers (Range, If-Match, etc.)
- * @return string Object's data
- * @throws InvalidResponseException unexpected response
- */
- function read($hdrs=array())
- {
- list($status, $reason, $data) =
- $this->container->cfs_http->get_object_to_string($this, $hdrs);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->read($hdrs);
- #}
- if (($status < 200) || ($status > 299
- && $status != 412 && $status != 304)) {
- throw new InvalidResponseException("Invalid response (".$status."): "
- . $this->container->cfs_http->get_error());
- }
- return $data;
- }
- /**
- * Streaming read of Object's data
- *
- * Given an open PHP resource (see PHP's fopen() method), fetch the Object's
- * data and write it to the open resource handle. This is useful for
- * streaming an Object's content to the browser (videos, images) or for
- * fetching content to a local file.
- *
- * Pass in $hdrs array to set specific custom HTTP headers such as
- * If-Match, If-None-Match, If-Modified-Since, Range, etc.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * # Assuming this is a web script to display the README to the
- * # user's browser:
- * #
- * <?php
- * // grab README from storage system
- * //
- * $my_docs = $conn->get_container("documents");
- * $doc = $my_docs->get_object("README");
- *
- * // Hand it back to user's browser with appropriate content-type
- * //
- * header("Content-Type: " . $doc->content_type);
- * $output = fopen("php://output", "w");
- * $doc->stream($output); # stream object content to PHP's output buffer
- * fclose($output);
- * ?>
- *
- * # See read() above for a more simple example.
- * #
- * </code>
- *
- * @param resource $fp open resource for writing data to
- * @param array $hdrs user-defined headers (Range, If-Match, etc.)
- * @return string Object's data
- * @throws InvalidResponseException unexpected response
- */
- function stream(&$fp, $hdrs=array())
- {
- list($status, $reason) =
- $this->container->cfs_http->get_object_to_stream($this,$fp,$hdrs);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->stream($fp, $hdrs);
- #}
- if (($status < 200) || ($status > 299
- && $status != 412 && $status != 304)) {
- throw new InvalidResponseException("Invalid response (".$status."): "
- .$reason);
- }
- return True;
- }
- /**
- * Store new Object metadata
- *
- * Write's an Object's metadata to the remote Object. This will overwrite
- * an prior Object metadata.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * $my_docs = $conn->get_container("documents");
- * $doc = $my_docs->get_object("README");
- *
- * # Define new metadata for the object
- * #
- * $doc->metadata = array(
- * "Author" => "EJ",
- * "Subject" => "How to use the PHP tests",
- * "Version" => "1.2.2"
- * );
- *
- * # Define additional headers for the object
- * #
- * $doc->headers = array(
- * "Content-Disposition" => "attachment",
- * );
- *
- * # Push the new metadata up to the storage system
- * #
- * $doc->sync_metadata();
- * </code>
- *
- * @return boolean <kbd>True</kbd> if successful, <kbd>False</kbd> otherwise
- * @throws InvalidResponseException unexpected response
- */
- function sync_metadata()
- {
- if (!empty($this->metadata) || !empty($this->headers) || $this->manifest) {
- $status = $this->container->cfs_http->update_object($this);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->sync_metadata();
- #}
- if ($status != 202) {
- throw new InvalidResponseException("Invalid response ("
- .$status."): ".$this->container->cfs_http->get_error());
- }
- return True;
- }
- return False;
- }
- /**
- * Store new Object manifest
- *
- * Write's an Object's manifest to the remote Object. This will overwrite
- * an prior Object manifest.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * $my_docs = $conn->get_container("documents");
- * $doc = $my_docs->get_object("README");
- *
- * # Define new manifest for the object
- * #
- * $doc->manifest = "container/prefix";
- *
- * # Push the new manifest up to the storage system
- * #
- * $doc->sync_manifest();
- * </code>
- *
- * @return boolean <kbd>True</kbd> if successful, <kbd>False</kbd> otherwise
- * @throws InvalidResponseException unexpected response
- */
- function sync_manifest()
- {
- return $this->sync_metadata();
- }
- /**
- * Upload Object's data to Cloud Files
- *
- * Write data to the remote Object. The $data argument can either be a
- * PHP resource open for reading (see PHP's fopen() method) or an in-memory
- * variable. If passing in a PHP resource, you must also include the $bytes
- * parameter.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * $my_docs = $conn->get_container("documents");
- * $doc = $my_docs->get_object("README");
- *
- * # Upload placeholder text in my README
- * #
- * $doc->write("This is just placeholder text for now...");
- * </code>
- *
- * @param string|resource $data string or open resource
- * @param float $bytes amount of data to upload (required for resources)
- * @param boolean $verify generate, send, and compare MD5 checksums
- * @return boolean <kbd>True</kbd> when data uploaded successfully
- * @throws SyntaxException missing required parameters
- * @throws BadContentTypeException if no Content-Type was/could be set
- * @throws MisMatchedChecksumException $verify is set and checksums unequal
- * @throws InvalidResponseException unexpected response
- */
- function write($data=NULL, $bytes=0, $verify=True)
- {
- if (!$data && !is_string($data)) {
- throw new SyntaxException("Missing data source.");
- }
- if ($bytes > MAX_OBJECT_SIZE) {
- throw new SyntaxException("Bytes exceeds maximum object size.");
- }
- if ($verify) {
- if (!$this->_etag_override) {
- $this->etag = $this->compute_md5sum($data);
- }
- } else {
- $this->etag = NULL;
- }
- $close_fh = False;
- if (!is_resource($data)) {
- # A hack to treat string data as a file handle. php://memory feels
- # like a better option, but it seems to break on Windows so use
- # a temporary file instead.
- #
- $fp = fopen("php://temp", "wb+");
- #$fp = fopen("php://memory", "wb+");
- fwrite($fp, $data, strlen($data));
- rewind($fp);
- $close_fh = True;
- $this->content_length = (float) strlen($data);
- if ($this->content_length > MAX_OBJECT_SIZE) {
- throw new SyntaxException("Data exceeds maximum object size");
- }
- $ct_data = substr($data, 0, 64);
- } else {
- $this->content_length = $bytes;
- $fp = $data;
- $ct_data = fread($data, 64);
- rewind($data);
- }
- $this->_guess_content_type($ct_data);
- list($status, $reason, $etag) =
- $this->container->cfs_http->put_object($this, $fp);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->write($data, $bytes, $verify);
- #}
- if ($status == 412) {
- if ($close_fh) { fclose($fp); }
- throw new SyntaxException("Missing Content-Type header");
- }
- if ($status == 422) {
- if ($close_fh) { fclose($fp); }
- throw new MisMatchedChecksumException(
- "Supplied and computed checksums do not match.");
- }
- if ($status != 201) {
- if ($close_fh) { fclose($fp); }
- throw new InvalidResponseException("Invalid response (".$status."): "
- . $this->container->cfs_http->get_error());
- }
- if (!$verify) {
- $this->etag = $etag;
- }
- if ($close_fh) { fclose($fp); }
- return True;
- }
- /**
- * Upload Object data from local filename
- *
- * This is a convenience function to upload the data from a local file. A
- * True value for $verify will cause the method to compute the Object's MD5
- * checksum prior to uploading.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * $my_docs = $conn->get_container("documents");
- * $doc = $my_docs->get_object("README");
- *
- * # Upload my local README's content
- * #
- * $doc->load_from_filename("/home/ej/cloudfiles/readme");
- * </code>
- *
- * @param string $filename full path to local file
- * @param boolean $verify enable local/remote MD5 checksum validation
- * @return boolean <kbd>True</kbd> if data uploaded successfully
- * @throws SyntaxException missing required parameters
- * @throws BadContentTypeException if no Content-Type was/could be set
- * @throws MisMatchedChecksumException $verify is set and checksums unequal
- * @throws InvalidResponseException unexpected response
- * @throws IOException error opening file
- */
- function load_from_filename($filename, $verify=True)
- {
- $fp = @fopen($filename, "r");
- if (!$fp) {
- throw new IOException("Could not open file for reading: ".$filename);
- }
- clearstatcache();
-
- $size = (float) sprintf("%u", filesize($filename));
- if ($size > MAX_OBJECT_SIZE) {
- throw new SyntaxException("File size exceeds maximum object size.");
- }
- $this->_guess_content_type($filename);
-
- $this->write($fp, $size, $verify);
- fclose($fp);
- return True;
- }
- /**
- * Save Object's data to local filename
- *
- * Given a local filename, the Object's data will be written to the newly
- * created file.
- *
- * Example:
- * <code>
- * # ... authentication/connection/container code excluded
- * # ... see previous examples
- *
- * # Whoops! I deleted my local README, let me download/save it
- * #
- * $my_docs = $conn->get_container("documents");
- * $doc = $my_docs->get_object("README");
- *
- * $doc->save_to_filename("/home/ej/cloudfiles/readme.restored");
- * </code>
- *
- * @param string $filename name of local file to write data to
- * @return boolean <kbd>True</kbd> if successful
- * @throws IOException error opening file
- * @throws InvalidResponseException unexpected response
- */
- function save_to_filename($filename)
- {
- $fp = @fopen($filename, "wb");
- if (!$fp) {
- throw new IOException("Could not open file for writing: ".$filename);
- }
- $result = $this->stream($fp);
- fclose($fp);
- return $result;
- }
- /**
- * Purge this Object from CDN Cache.
- * Example:
- * <code>
- * # ... authentication code excluded (see previous examples) ...
- * #
- * $conn = new CF_Authentication($auth);
- * $container = $conn->get_container("cdn_enabled");
- * $obj = $container->get_object("object");
- * $obj->purge_from_cdn("user@domain.com");
- * # or
- * $obj->purge_from_cdn();
- * # or
- * $obj->purge_from_cdn("user1@domain.com,user2@domain.com");
- * @returns boolean True if successful
- * @throws CDNNotEnabledException if CDN Is not enabled on this connection
- * @throws InvalidResponseException if the response expected is not returned
- */
- function purge_from_cdn($email=null)
- {
- if (!$this->container->cfs_http->getCDNMUrl())
- {
- throw new CDNNotEnabledException(
- "Authentication response did not indicate CDN availability");
- }
- $status = $this->container->cfs_http->purge_from_cdn($this->container->name . "/" . $this->name, $email);
- if ($status < 199 or $status > 299) {
- throw new InvalidResponseException(
- "Invalid response (".$status."): ".$this->container->cfs_http->get_error());
- }
- return True;
- }
- /**
- * Set Object's MD5 checksum
- *
- * Manually set the Object's ETag. Including the ETag is mandatory for
- * Cloud Files to perform end-to-end verification. Omitting the ETag forces
- * the user to handle any data integrity checks.
- *
- * @param string $etag MD5 checksum hexidecimal string
- */
- function set_etag($etag)
- {
- $this->etag = $etag;
- $this->_etag_override = True;
- }
- /**
- * Object's MD5 checksum
- *
- * Accessor method for reading Object's private ETag attribute.
- *
- * @return string MD5 checksum hexidecimal string
- */
- function getETag()
- {
- return $this->etag;
- }
- /**
- * Compute the MD5 checksum
- *
- * Calculate the MD5 checksum on either a PHP resource or data. The argument
- * may either be a local filename, open resource for reading, or a string.
- *
- * <b>WARNING:</b> if you are uploading a big file over a stream
- * it could get very slow to compute the md5 you probably want to
- * set the $verify parameter to False in the write() method and
- * compute yourself the md5 before if you have it.
- *
- * @param filename|obj|string $data filename, open resource, or string
- * @return string MD5 checksum hexidecimal string
- */
- function compute_md5sum(&$data)
- {
- if (function_exists("hash_init") && is_resource($data)) {
- $ctx = hash_init('md5');
- while (!feof($data)) {
- $buffer = fgets($data, 65536);
- hash_update($ctx, $buffer);
- }
- $md5 = hash_final($ctx, false);
- rewind($data);
- } elseif ((string)is_file($data)) {
- $md5 = md5_file($data);
- } else {
- $md5 = md5($data);
- }
- return $md5;
- }
- /**
- * PRIVATE: fetch information about the remote Object if it exists
- */
- private function _initialize()
- {
- list($status, $reason, $etag, $last_modified, $content_type,
- $content_length, $metadata, $manifest, $headers) =
- $this->container->cfs_http->head_object($this);
- #if ($status == 401 && $this->_re_auth()) {
- # return $this->_initialize();
- #}
- if ($status == 404) {
- return False;
- }
- if ($status < 200 || $status > 299) {
- throw new InvalidResponseException("Invalid response (".$status."): "
- . $this->container->cfs_http->get_error());
- }
- $this->etag = $etag;
- $this->last_modified = $last_modified;
- $this->content_type = $content_type;
- $this->content_length = $content_length;
- $this->metadata = $metadata;
- $this->headers = $headers;
- $this->manifest = $manifest;
- return True;
- }
- #private function _re_auth()
- #{
- # $new_auth = new CF_Authentication(
- # $this->cfs_auth->username,
- # $this->cfs_auth->api_key,
- # $this->cfs_auth->auth_host,
- # $this->cfs_auth->account);
- # $new_auth->authenticate();
- # $this->container->cfs_auth = $new_auth;
- # $this->container->cfs_http->setCFAuth($this->cfs_auth);
- # return True;
- #}
- }
- /* vim: set expandtab tabstop=4 shiftwidth=4 softtabstop=4: */
- /*
- * Local variables:
- * tab-width: 4
- * c-basic-offset: 4
- * c-hanging-comment-ender-p: nil
- * End:
- */
- ?>
|