123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529 |
- <?php
- namespace 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 = <<<BBB
- 0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
- 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);
- }
- }
- }
|