| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529 | <?phpnamespace OSS\Core;/** * Class OssUtil * * Oss Util class for OssClient. The caller could use it for formating the result from OssClient. * * @package OSS */class OssUtil{    const OSS_CONTENT = 'content';    const OSS_LENGTH = 'length';    const OSS_HEADERS = 'headers';    const OSS_MAX_OBJECT_GROUP_VALUE = 1000;    const OSS_MAX_PART_SIZE = 5368709120; // 5GB    const OSS_MID_PART_SIZE = 10485760; // 10MB    const OSS_MIN_PART_SIZE = 102400; // 100KB    /**     * Generate query params     *     * @param array $options: a key-value pair array.     * @return string: the key-value list in the format such as key1=value1&key2=value2     */    public static function toQueryString($options = array())    {        $temp = array();        uksort($options, 'strnatcasecmp');        foreach ($options as $key => $value) {            if (is_string($key) && !is_array($value)) {                $temp[] = rawurlencode($key) . '=' . rawurlencode($value);            }        }        return implode('&', $temp);    }    /**     * Html encoding '<', '>', '&', '\', '"' in subject parameter.      *     * @param string $subject     * @return string     */    public static function sReplace($subject)    {        $search = array('<', '>', '&', '\'', '"');        $replace = array('<', '>', '&', ''', '"');        return str_replace($search, $replace, $subject);    }    /**     * Check whether the string includes any chinese character     *     * @param $str     * @return int     */    public static function chkChinese($str)    {        return preg_match('/[\x80-\xff]./', $str);    }    /**     * Checks if the string is encoded by GB2312.     *     * @param string $str     * @return boolean false UTF-8 encoding  TRUE GB2312 encoding     */    public static function isGb2312($str)    {        for ($i = 0; $i < strlen($str); $i++) {            $v = ord($str[$i]);            if ($v > 127) {                if (($v >= 228) && ($v <= 233)) {                    if (($i + 2) >= (strlen($str) - 1)) return true;  // not enough characters                    $v1 = ord($str[$i + 1]);                    $v2 = ord($str[$i + 2]);                    if (($v1 >= 128) && ($v1 <= 191) && ($v2 >= 128) && ($v2 <= 191))                        return false;                    else                        return true;                }            }        }        return false;    }    /**     * Checks if the string is encoded by GBK     *     * @param string $str     * @param boolean $gbk     * @return boolean     */    public static function checkChar($str, $gbk = true)    {        for ($i = 0; $i < strlen($str); $i++) {            $v = ord($str[$i]);            if ($v > 127) {                if (($v >= 228) && ($v <= 233)) {                    if (($i + 2) >= (strlen($str) - 1)) return $gbk ? true : FALSE;  // not enough characters                    $v1 = ord($str[$i + 1]);                    $v2 = ord($str[$i + 2]);                    if ($gbk) {                        return (($v1 >= 128) && ($v1 <= 191) && ($v2 >= 128) && ($v2 <= 191)) ? FALSE : TRUE;//GBK                    } else {                        return (($v1 >= 128) && ($v1 <= 191) && ($v2 >= 128) && ($v2 <= 191)) ? TRUE : FALSE;                    }                }            }        }        return $gbk ? TRUE : FALSE;    }    /**     * Checks if the bucket name is valid     * bucket naming rules     * 1. Can only include lowercase letters, numbers, or dashes     * 2. Must start and end with lowercase letters or numbers     * 3. Must be within a length from 3 to 63 bytes.     *     * @param string $bucket Bucket name     * @return boolean     */    public static function validateBucket($bucket)    {        $pattern = '/^[a-z0-9][a-z0-9-]{2,62}$/';        if (!preg_match($pattern, $bucket)) {            return false;        }        return true;    }    /**     * Checks if object name is valid     * object naming rules:     * 1. Must be within a length from 1 to 1023 bytes     * 2. Cannot start with '/' or '\\'.     * 3. Must be encoded in UTF-8.     *     * @param string $object Object名称     * @return boolean     */    public static function validateObject($object)    {        $pattern = '/^.{1,1023}$/';        if (!preg_match($pattern, $object) ||            self::startsWith($object, '/') || self::startsWith($object, '\\')        ) {            return false;        }        return true;    }    /**     * Checks if $str starts with $findMe     *     * @param string $str     * @param string $findMe     * @return bool     */    public static function startsWith($str, $findMe)    {        if (strpos($str, $findMe) === 0) {            return true;        } else {            return false;        }    }    /**     * Generate the xml message of createBucketXmlBody.     *     * @param string $storageClass     * @return string     */    public static function createBucketXmlBody($storageClass)    {        $xml = new \SimpleXMLElement('<?xml version="1.0" encoding="UTF-8"?><CreateBucketConfiguration></CreateBucketConfiguration>');        $xml->addChild('StorageClass',  $storageClass);        return $xml->asXML();    }    /**     * validate $options     *     * @param array $options     * @throws OssException     * @return boolean     */    public static function validateOptions($options)    {        //$options        if ($options != NULL && !is_array($options)) {            throw new OssException ($options . ':' . 'option must be array');        }    }    /**     * check whether the Content is valid.     *     * @param $content string     * @throws OssException     */    public static function validateContent($content)    {        if (empty($content)) {            throw new OssException("http body content is invalid");        }    }    /**     * Check if BUCKET/OBJECT/OBJECT GROUP is empty.     *     * @param  string $name     * @param  string $errMsg     * @throws OssException     * @return void     */    public static function throwOssExceptionWithMessageIfEmpty($name, $errMsg)    {        if (empty($name)) {            if (is_string($name) && $name == '0')                return;            throw new OssException($errMsg);        }    }    /**     * This is a method for test only. DO NOT USE.     *     * @param $filename     * @param $size     */    public static function generateFile($filename, $size)    {        if (file_exists($filename) && $size == filesize($filename)) {            echo $filename . " already exists, no need to create again. ";            return;        }        $part_size = 1 * 1024 * 1024;        $fp = fopen($filename, "w");        $characters = <<<BBB0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';BBB;        $charactersLength = strlen($characters);        if ($fp) {            while ($size > 0) {                if ($size < $part_size) {                    $write_size = $size;                } else {                    $write_size = $part_size;                }                $size -= $write_size;                $a = $characters[rand(0, $charactersLength - 1)];                $content = str_repeat($a, $write_size);                $flag = fwrite($fp, $content);                if (!$flag) {                    echo "write to " . $filename . " failed. <br>";                    break;                }            }        } else {            echo "open " . $filename . " failed. <br>";        }        fclose($fp);    }    /**     * Get MD5 of the file.     *     * @param $filename     * @param $from_pos     * @param $to_pos     * @return string     */    public static function getMd5SumForFile($filename, $from_pos, $to_pos)    {        $content_md5 = "";        if (($to_pos - $from_pos) > self::OSS_MAX_PART_SIZE) {            return $content_md5;        }        $filesize = filesize($filename);        if ($from_pos >= $filesize || $to_pos >= $filesize || $from_pos < 0 || $to_pos < 0) {            return $content_md5;        }        $total_length = $to_pos - $from_pos + 1;        $buffer = 8192;        $left_length = $total_length;        if (!file_exists($filename)) {            return $content_md5;        }        if (false === $fh = fopen($filename, 'rb')) {            return $content_md5;        }        fseek($fh, $from_pos);        $data = '';        while (!feof($fh)) {            if ($left_length >= $buffer) {                $read_length = $buffer;            } else {                $read_length = $left_length;            }            if ($read_length <= 0) {                break;            } else {                $data .= fread($fh, $read_length);                $left_length = $left_length - $read_length;            }        }        fclose($fh);        $content_md5 = base64_encode(md5($data, true));        return $content_md5;    }    /**     * Check if the OS is Windows. The default encoding in Windows is GBK.     *     * @return bool     */    public static function isWin()    {        return strtoupper(substr(PHP_OS, 0, 3)) == "WIN";    }    /**     * Encodes the file path from GBK to UTF-8.     * The default encoding in Windows is GBK.      * And if the file path is in Chinese, the file would not be found without the transcoding to UTF-8.     *     * @param $file_path     * @return string     */    public static function encodePath($file_path)    {        if (self::chkChinese($file_path) && self::isWin()) {            $file_path = iconv('utf-8', 'gbk', $file_path);        }        return $file_path;    }    /**     * Check if the endpoint is in the IPv4 format, such as xxx.xxx.xxx.xxx:port or xxx.xxx.xxx.xxx.     *     * @param string $endpoint The endpoint to check.     * @return boolean     */    public static function isIPFormat($endpoint)    {        $ip_array = explode(":", $endpoint);        $hostname = $ip_array[0];        $ret = filter_var($hostname, FILTER_VALIDATE_IP);        if (!$ret) {            return false;        } else {            return true;        }    }    /**     * Get the host:port from endpoint.     *     * @param string $endpoint the endpoint.     * @return boolean     */    public static function getHostPortFromEndpoint($endpoint)    {        $str = $endpoint;        $pos = strpos($str, "://");        if ($pos !== false) {            $str = substr($str, $pos+3);        }            $pos = strpos($str, '#');        if ($pos !== false) {            $str = substr($str, 0, $pos);        }            $pos = strpos($str, '?');        if ($pos !== false) {            $str = substr($str, 0, $pos);        }            $pos = strpos($str, '/');        if ($pos !== false) {            $str = substr($str, 0, $pos);        }            $pos = strpos($str, '@');        if ($pos !== false) {            $str = substr($str, $pos+1);        }               if (!preg_match('/^[\w.-]+(:[0-9]+)?$/', $str)) {            throw new OssException("endpoint is invalid:" . $endpoint);        }        return $str;    }    /**     * Generate the xml message of DeleteMultiObjects.     *     * @param string[] $objects     * @param bool $quiet     * @return string     */    public static function createDeleteObjectsXmlBody($objects, $quiet)    {        $xml = new \SimpleXMLElement('<?xml version="1.0" encoding="utf-8"?><Delete></Delete>');        $xml->addChild('Quiet', $quiet);        foreach ($objects as $object) {            $sub_object = $xml->addChild('Object');            $object = OssUtil::sReplace($object);            $sub_object->addChild('Key', $object);        }        return $xml->asXML();    }    /**     * Generate the xml message of DeleteMultiObjects.     *     * @param DeleteObjectInfo[] $objects     * @param bool $quiet     * @return string     */    public static function createDeleteObjectVersionsXmlBody($objects, $quiet)    {        $xml = new \SimpleXMLElement('<?xml version="1.0" encoding="utf-8"?><Delete></Delete>');        $xml->addChild('Quiet', $quiet);        foreach ($objects as $object) {            $sub_object = $xml->addChild('Object');            $key = OssUtil::sReplace($object->getKey());            $sub_object->addChild('Key', $key);            if (!empty($object->getVersionId())) {                $sub_object->addChild('VersionId', $object->getVersionId());            }        }        return $xml->asXML();    }    /**     * Generate the xml message of CompleteMultipartUpload.     *     * @param array[] $listParts     * @return string     */    public static function createCompleteMultipartUploadXmlBody($listParts)    {        $xml = new \SimpleXMLElement('<?xml version="1.0" encoding="utf-8"?><CompleteMultipartUpload></CompleteMultipartUpload>');        foreach ($listParts as $node) {            $part = $xml->addChild('Part');            $part->addChild('PartNumber', $node['PartNumber']);            $part->addChild('ETag', $node['ETag']);        }        return $xml->asXML();    }    /**     * Read the directory, return a associative array in which the MD5 is the named key and the <path,filanme> is the value.     *     * @param string $dir     * @param string $exclude     * @param bool $recursive     * @return string[]     */    public static function readDir($dir, $exclude = ".|..|.svn|.git", $recursive = false)    {        $file_list_array = array();        $base_path = $dir;        $exclude_array = explode("|", $exclude);        $exclude_array = array_unique(array_merge($exclude_array, array('.', '..')));        if ($recursive) {            foreach (new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($dir)) as $new_file) {                if ($new_file->isDir()) continue;                $object = str_replace($base_path, '', $new_file);                if (!in_array(strtolower($object), $exclude_array)) {                    $object = ltrim($object, '/');                    if (is_file($new_file)) {                        $key = md5($new_file . $object, false);                        $file_list_array[$key] = array('path' => $new_file, 'file' => $object,);                    }                }            }        } else if ($handle = opendir($dir)) {            while (false !== ($file = readdir($handle))) {                if (!in_array(strtolower($file), $exclude_array)) {                    $new_file = $dir . '/' . $file;                    $object = $file;                    $object = ltrim($object, '/');                    if (is_file($new_file)) {                        $key = md5($new_file . $object, false);                        $file_list_array[$key] = array('path' => $new_file, 'file' => $object,);                    }                }            }            closedir($handle);        }        return $file_list_array;    }    /**     * Decode key based on the encoding type     *     * @param string $key     * @param string $encoding     * @return string     */    public static function decodeKey($key, $encoding)    {        if ($encoding == "") {            return $key;        }        if ($encoding == "url") {            return rawurldecode($key);        } else {            throw new OssException("Unrecognized encoding type: " . $encoding);        }    }}
 |