• FastDFS php API


      1 <?php
      2 
      3 if (!class_exists('FastDFS', false)) {
      4     define('FDFS_PROTO_PKG_LEN_SIZE', 8);
      5     define('FDFS_PROTO_CMD_ACTIVE_TEST', 111);
      6     define('FDFS_PROTO_CMD_RESP', 100);
      7     define('FDFS_PROTO_CMD_UPLOAD_SLAVE_FILE', 21);
      8     define('FDFS_PROTO_CMD_DELETE_FILE', 12);
      9     define('FDFS_PROTO_CMD_GET_METADATA', 15);
     10     define('FDFS_PROTO_CMD_SET_METADATA', 13);
     11 
     12     //body_length + command + status
     13     define('FDFS_HEADER_LENGTH', 10);
     14     define('FDFS_IP_ADDRESS_SIZE', 16);
     15     define('FDFS_FILE_EXT_NAME_MAX_LEN', 6);
     16     define('FDFS_GROUP_NAME_MAX_LEN', 16);
     17     define('FDFS_OVERWRITE_METADATA', 1);
     18     define('FDFS_MERGE_METADATA', 2);
     19 
     20     // 连接超时时间
     21     define('FDFS_CONNECT_TIME_OUT', 5);
     22     define('FDFS_FILE_PREFIX_MAX_LEN', 16);
     23 
     24     //传输超时时间
     25     define('FDFS_TRANSFER_TIME_OUT', 0);
     26     define('FDFS_QUERY_STORE_WITHOUT_GROUP_ONE', 101);
     27     define('FDFS_QUERY_STORE_WITH_GROUP_ONE', 104);
     28 
     29     define('FDFS_TRACKER_QUERY_STORAGE_STORE_BODY_LEN', FDFS_GROUP_NAME_MAX_LEN + FDFS_IP_ADDRESS_SIZE + FDFS_PROTO_PKG_LEN_SIZE);
     30 
     31     class FastDFS {
     32 
     33         public $gConfig = array();
     34 
     35         /**
     36          *
     37          * @var FastDFSTrackerClient
     38          */
     39         private $tracker;
     40 
     41         /**
     42          *
     43          * @var FastDFSStorageClient
     44          */
     45         private $storage;
     46         private $error = array(
     47             'code' => 0,
     48             'msg' => ''
     49         );
     50 
     51         /**
     52          * 要使用这个类,你一定要在php的ini文件中进行fastdfs的配置
     53          * 
     54          * @throws FastDFSException
     55          */
     56         public function __construct() {
     57             $configFile = '';
     58             $ini = parse_ini_file(php_ini_loaded_file());
     59 
     60             if (!isset($ini['fastdfs_client.tracker_group_count'])) {
     61                 throw new FastDFSException("no define fastdfs config");
     62             }
     63             for ($i = 0; $i < $ini['fastdfs_client.tracker_group_count']; $i++) {
     64                 if (isset($ini['fastdfs_client.tracker_group' . $i])) {
     65                     $configFile = $ini['fastdfs_client.tracker_group' . $i];
     66                     break;
     67                 }
     68             }
     69             if (!file_exists($configFile)) {
     70                 throw new FastDFSException("client config $configFile not found");
     71             }
     72             $this->gConfig = parse_ini_file($configFile);
     73             list($this->gConfig['tracker_host'], $this->gConfig['tracker_port']) = explode(':', $this->gConfig['tracker_server']);
     74         }
     75 
     76         /**
     77          * 获得一个tracker
     78          * 
     79          * @return FastDFSTrackerClient
     80          */
     81         public function tracker_get_connection() {
     82             $this->tracker = new FastDFSTrackerClient($this, $this->gConfig['tracker_host'], $this->gConfig['tracker_port']);
     83 
     84             return $this->tracker;
     85         }
     86 
     87         /**
     88          * 通过tracker获取一个stroage
     89          * 
     90          * @param string $groupName 文件组名,当为空时,组名由tracker决定
     91          * @param FastDFSTrackerClient $tracker
     92          * @return FastDFSStorageClient
     93          */
     94         public function tracker_query_storage_store($groupName, FastDFSTrackerClient $tracker) {
     95             $this->storage = new FastDFSStorageClient($this, $groupName, $tracker);
     96 
     97             return $this->storage;
     98         }
     99 
    100         /**
    101          * 测试一下tracker服务器是否正常
    102          * 
    103          * @param FastDFSTrackerClient $tracker
    104          * @return boolean
    105          */
    106         public function active_test(FastDFSTrackerClient $tracker = null) {
    107             $this->initTrackerAndStorage($tracker);
    108             
    109             $header = self::packHeader(FDFS_PROTO_CMD_ACTIVE_TEST, 0);
    110             $tracker->send($header);
    111 
    112             $resHeader = self::parseHeader($tracker->read(FDFS_HEADER_LENGTH));
    113 
    114             return $resHeader['status'] == 0 ? true : false;
    115         }
    116 
    117         public function get_last_error_no() {
    118             return $this->error['code'];
    119         }
    120 
    121         public function add_error($errorNo, $info) {
    122             $this->error['code'] = $errorNo;
    123             $this->error['msg'] = $info;
    124         }
    125 
    126         public function get_last_error_info() {
    127             return $this->error['msg'];
    128         }
    129 
    130         /**
    131          * 在storage中删除一个文件
    132          * 
    133          * @param string $groupName 文件所在的组名
    134          * @param string $remoteFile 要删除的文件路径
    135          * @param FastDFSStorageClient $tracker
    136          * @param FastDFSStorageClient $storage
    137          */
    138         public function storage_delete_file($groupName, $remoteFile, FastDFSStorageClient $tracker, FastDFSStorageClient $storage) {
    139             $this->initTrackerAndStorage($tracker, $storage, $groupName);
    140 
    141             $this->storage->deleteFile($groupName, $remoteFile);
    142         }
    143 
    144         /**
    145          * 往storage中上传一个文件
    146          * 
    147          * @param string $localFile 你本地的文件路径
    148          * @param string $extName 文件的扩展名,当名优提供扩展名时,会自动取文件的扩展名
    149          * @param array $metas 文件的附加信息
    150          * @param string $groupName 所在的组名,可以为空,为空时,由tracker决定
    151          * @param FastDFSTrackerClient $tracker
    152          * @param FastDFSStorageClient $storage
    153          */
    154         public function storage_upload_by_filename($localFile, $extName = '', $metas = array(), $groupName = '', FastDFSTrackerClient $tracker = null, FastDFSStorageClient $storage = null) {
    155             $this->initTrackerAndStorage($tracker, $storage, $groupName);
    156 
    157             return $this->storage->uploadByFilename($localFile, $extName, $metas);
    158         }
    159 
    160         /**
    161          * 上传一个文件的附属文件,主要使用一个图片有缩略图的情况下
    162          * 
    163          * @param string $localFile 本地文件的路径,缩略图的文件路径
    164          * @param string $groupName 组名,最好和主文件在同一个组
    165          * @param string $masterFileName 主文件名
    166          * @param string $prefix 文件的前缀
    167          * @param string $extName 文件的后缀,可以为空,为空时,由tracker决定
    168          * @param array $meta 附件信息
    169          * @param FastDFSTrackerClient $tracker
    170          * @param FastDFSStorageClient $storage
    171          */
    172         public function storage_upload_slave_by_filename($localFile, $groupName, $masterFileName, $prefix = '', $extName = '', $meta = array(), FastDFSTrackerClient $tracker = null, FastDFSStorageClient $storage = null) {
    173             $this->initTrackerAndStorage($tracker, $storage, $groupName);
    174             /*echo $localFile."<br/>".$groupName."<br/>".$masterFileName."<br/>".$prefix;
    175             exit;*/
    176             return $this->storage->uploadSalveFile($localFile, $groupName, $masterFileName, $prefix, $extName, $meta);
    177         }
    178 
    179         /**
    180          * 检查这个文件是否已经存在
    181          * 
    182          * @param string $groupName 文件所在组名
    183          * @param string $remoteFile 文件在storage中的名字
    184          * @param FastDFSStorageClient $tracker
    185          * @param FastDFSStorageClient $storage
    186          */
    187         public function storage_file_exist($groupName, $remoteFile, FastDFSTrackerClient $tracker, FastDFSStorageClient $storage) {
    188             $this->initTrackerAndStorage($tracker, $storage, $groupName);
    189 
    190             return $this->storage->fileExists($groupName, $remoteFile);
    191         }
    192 
    193         public function close() {
    194             if ($this->tracker) {
    195                 $this->tracker->close();
    196                 $this->tracker = null;
    197             }
    198         }
    199 
    200         public function tracker_close_all_connections() {
    201             $this->close();
    202             if (!$this->storage) {
    203                 $this->storage->close();
    204             }
    205         }
    206 
    207         public static function padding($str, $len) {
    208 
    209             $str_len = strlen($str);
    210 
    211             return $str_len > $len ? substr($str, 0, $len) : $str . pack('x' . ($len - $str_len));
    212         }
    213 
    214         /**
    215          * 
    216          * @param int $command
    217          * @param int $length
    218          * @return bytes
    219          */
    220         public static function packHeader($command, $length = 0) {
    221             return self::packU64($length) . pack('Cx', $command);
    222         }
    223 
    224         public static function packMetaData($data) {
    225             $S1 = "x01";
    226             $S2 = "x02";
    227 
    228             $list = array();
    229             foreach ($data as $key => $val) {
    230                 $list[] = $key . $S2 . $val;
    231             };
    232 
    233             return implode($S1, $list);
    234         }
    235 
    236         public static function parseMetaData($data) {
    237 
    238             $S1 = "x01";
    239             $S2 = "x02";
    240 
    241             $arr = explode($S1, $data);
    242             $result = array();
    243 
    244             foreach ($arr as $val) {
    245                 list($k, $v) = explode($S2, $val);
    246                 $result[$k] = $v;
    247             }
    248 
    249             return $result;
    250         }
    251 
    252         public static function parseHeader($str, $len = FDFS_HEADER_LENGTH) {
    253 
    254             assert(strlen($str) === $len);
    255 
    256             $result = unpack('C10', $str);
    257 
    258             $length = self::unpackU64(substr($str, 0, 8));
    259             $command = $result[9];
    260             $status = $result[10];
    261 
    262             return array(
    263                 'length' => $length,
    264                 'command' => $command,
    265                 'status' => $status
    266             );
    267         }
    268 
    269         /**
    270          * From: sphinxapi.php
    271          */
    272         private static function unpackU64($v) {
    273             list ( $hi, $lo ) = array_values(unpack("N*N*", $v));
    274 
    275             if (PHP_INT_SIZE >= 8) {
    276                 if ($hi < 0)
    277                     $hi += (1 << 32); // because php 5.2.2 to 5.2.5 is totally fucked up again
    278                 if ($lo < 0)
    279                     $lo += (1 << 32);
    280 
    281                 // x64, int
    282                 if ($hi <= 2147483647)
    283                     return ($hi << 32) + $lo;
    284 
    285                 // x64, bcmath
    286                 if (function_exists("bcmul"))
    287                     return bcadd($lo, bcmul($hi, "4294967296"));
    288 
    289                 // x64, no-bcmath
    290                 $C = 100000;
    291                 $h = ((int) ($hi / $C) << 32) + (int) ($lo / $C);
    292                 $l = (($hi % $C) << 32) + ($lo % $C);
    293                 if ($l > $C) {
    294                     $h += (int) ($l / $C);
    295                     $l = $l % $C;
    296                 }
    297 
    298                 if ($h == 0)
    299                     return $l;
    300                 return sprintf("%d%05d", $h, $l);
    301             }
    302 
    303             // x32, int
    304             if ($hi == 0) {
    305                 if ($lo > 0)
    306                     return $lo;
    307                 return sprintf("%u", $lo);
    308             }
    309 
    310             $hi = sprintf("%u", $hi);
    311             $lo = sprintf("%u", $lo);
    312 
    313             // x32, bcmath
    314             if (function_exists("bcmul"))
    315                 return bcadd($lo, bcmul($hi, "4294967296"));
    316 
    317             // x32, no-bcmath
    318             $hi = (float) $hi;
    319             $lo = (float) $lo;
    320 
    321             $q = floor($hi / 10000000.0);
    322             $r = $hi - $q * 10000000.0;
    323             $m = $lo + $r * 4967296.0;
    324             $mq = floor($m / 10000000.0);
    325             $l = $m - $mq * 10000000.0;
    326             $h = $q * 4294967296.0 + $r * 429.0 + $mq;
    327 
    328             $h = sprintf("%.0f", $h);
    329             $l = sprintf("%07.0f", $l);
    330             if ($h == "0")
    331                 return sprintf("%.0f", (float) $l);
    332             return $h . $l;
    333         }
    334 
    335         private function initTrackerAndStorage(FastDFSTrackerClient $tracker = null, FastDFSStorageClient $storage = null, $groupName = '') {
    336             $reNewStorage = false;
    337             if ($tracker && $tracker !== $this->tracker) {
    338                 $this->tracker_get_connection();
    339             }
    340             if (($storage && $storage !== $this->storage) || $reNewStorage) {
    341                 $this->tracker_query_storage_store($groupName, $this->tracker);
    342             }
    343         }
    344 
    345         /**
    346          * From: sphinxapi.php
    347          */
    348         public static function packU64($v) {
    349 
    350 
    351             assert(is_numeric($v));
    352 
    353             // x64
    354             if (PHP_INT_SIZE >= 8) {
    355                 assert($v >= 0);
    356 
    357                 // x64, int
    358                 if (is_int($v))
    359                     return pack("NN", $v >> 32, $v & 0xFFFFFFFF);
    360 
    361                 // x64, bcmath
    362                 if (function_exists("bcmul")) {
    363                     $h = bcdiv($v, 4294967296, 0);
    364                     $l = bcmod($v, 4294967296);
    365                     return pack("NN", $h, $l);
    366                 }
    367 
    368                 // x64, no-bcmath
    369                 $p = max(0, strlen($v) - 13);
    370                 $lo = (int) substr($v, $p);
    371                 $hi = (int) substr($v, 0, $p);
    372 
    373                 $m = $lo + $hi * 1316134912;
    374                 $l = $m % 4294967296;
    375                 $h = $hi * 2328 + (int) ($m / 4294967296);
    376 
    377                 return pack("NN", $h, $l);
    378             }
    379 
    380             // x32, int
    381             if (is_int($v))
    382                 return pack("NN", 0, $v);
    383 
    384             // x32, bcmath
    385             if (function_exists("bcmul")) {
    386                 $h = bcdiv($v, "4294967296", 0);
    387                 $l = bcmod($v, "4294967296");
    388                 return pack("NN", (float) $h, (float) $l); // conversion to float is intentional; int would lose 31st bit
    389             }
    390 
    391             // x32, no-bcmath
    392             $p = max(0, strlen($v) - 13);
    393             $lo = (float) substr($v, $p);
    394             $hi = (float) substr($v, 0, $p);
    395 
    396             $m = $lo + $hi * 1316134912.0;
    397             $q = floor($m / 4294967296.0);
    398             $l = $m - ($q * 4294967296.0);
    399             $h = $hi * 2328.0 + $q;
    400 
    401             return pack("NN", $h, $l);
    402         }
    403 
    404     }
    405 
    406     abstract class FastDFSBase {
    407 
    408         abstract public function getSocket();
    409 
    410         abstract public function close();
    411 
    412         public function read($length, $socket = null) {
    413             if (!$socket) {
    414                 $socket = $this->getSocket();
    415             }
    416 
    417             if (feof($socket)) {
    418                 throw new FastDFS_Exception('connection unexpectedly closed (timed out?)', $this->_errno);
    419             }
    420 
    421             $data = stream_get_contents($socket, $length);
    422 
    423             assert($length === strlen($data));
    424 
    425             return $data;
    426         }
    427 
    428         public function send($data, $length = 0, $socket = null) {
    429             if (!$socket) {
    430                 $socket = $this->getSocket();
    431             }
    432 
    433             if (!$length) {
    434                 $length = strlen($data);
    435             }
    436 
    437             if (feof($socket) || fwrite($socket, $data, $length) !== $length) {
    438                 throw new Exception('connection unexpectedly closed (timed out?)');
    439             }
    440 
    441             return true;
    442         }
    443 
    444     }
    445 
    446     class FastDFSTrackerClient extends FastDFSBase {
    447 
    448         private $host;
    449         private $port;
    450 
    451         /**
    452          *
    453          * @var FastDFS
    454          */
    455         private $dfs;
    456         private $_socket;
    457 
    458         public function __construct(FastDFS &$dfs, $host, $port) {
    459             $this->host = $host;
    460             $this->port = $port;
    461             $this->dfs = $dfs;
    462 
    463             $this->_socket = @fsockopen("tcp://$host", $port, $errno, $errstr, $this->dfs->gConfig['connect_timeout']);
    464             if (!$this->_socket) {
    465                 $this->dfs->add_error(-2, $errstr);
    466             }
    467         }
    468 
    469         public function getSocket() {
    470             return $this->_socket;
    471         }
    472 
    473         public function close() {
    474             fclose($this->_socket);
    475         }
    476 
    477     }
    478 
    479     class FastDFSStorageClient extends FastDFSBase {
    480 
    481         private $groupName;
    482 
    483         /**
    484          *
    485          * @var FastDFSTrackerClient 
    486          */
    487         private $tracker;
    488 
    489         /**
    490          *
    491          * @var FastDFS
    492          */
    493         private $dfs;
    494         private $_socket;
    495         private $host;
    496         private $port;
    497         private $storeIndex;
    498 
    499         public function __construct(FastDFS &$dfs, $groupName, FastDFSTrackerClient $tracker) {
    500             $this->tracker = $tracker;
    501             $this->dfs = $dfs;
    502 
    503             $reqBody = '';
    504             if ($groupName) {
    505                 $cmd = FDFS_QUERY_STORE_WITH_GROUP_ONE;
    506                 $len = FDFS_GROUP_NAME_MAX_LEN;
    507                 $reqBody = FastDFS::padding($groupName, $len);
    508             } else {
    509                 $cmd = FDFS_QUERY_STORE_WITHOUT_GROUP_ONE;
    510                 $len = 0;
    511             }
    512             $reqHeader = FastDFS::packHeader($cmd, $len);
    513             $this->tracker->send($reqHeader . $reqBody);
    514 
    515             $resHeader = $this->tracker->read(FDFS_HEADER_LENGTH);
    516             $resInfo = FastDFS::parseHeader($resHeader);
    517 
    518             if ($resInfo['status'] != 0) {
    519                 throw new Exception("something wrong with get storage by group name", $resInfo['status']);
    520             }
    521 
    522             $resBody = !!$resInfo['length'] ? $this->tracker->read($resInfo['length']) : '';
    523             $this->groupName = trim(substr($resBody, 0, FDFS_GROUP_NAME_MAX_LEN));
    524             $this->host = trim(substr($resBody, FDFS_GROUP_NAME_MAX_LEN, FDFS_IP_ADDRESS_SIZE + 1));
    525 
    526             list(,, $this->port) = unpack('N2', substr($resBody, FDFS_GROUP_NAME_MAX_LEN + FDFS_IP_ADDRESS_SIZE - 1, FDFS_PROTO_PKG_LEN_SIZE));
    527 
    528             $this->storeIndex = ord(substr($resBody, -1));
    529 
    530             $this->_socket = @fsockopen($this->host, $this->port, $errno, $errstr, $this->dfs->gConfig['connect_timeout']);
    531 
    532             if (!$this->_socket) {
    533                 $this->dfs->add_error($errno, $errstr);
    534             }
    535         }
    536 
    537         public function getSocket() {
    538             return $this->_socket;
    539         }
    540 
    541         public function getStorePathIndex() {
    542             return $this->storeIndex;
    543         }
    544 
    545         public function uploadByFilename($localFile, $extName, $metas) {
    546             if (!file_exists($localFile)) {
    547                 throw new FastDFSException("$localFile file is not exists");
    548             }
    549             $pathInfo = pathinfo($localFile);
    550 
    551             $extName = $extName ? $extName : $pathInfo['extension'];
    552             $extLen = strlen($extName);
    553 
    554             if ($extLen > FDFS_FILE_EXT_NAME_MAX_LEN) {
    555                 throw new FastDFSException("file ext too long");
    556             }
    557             $fp = fopen($localFile, 'rb');
    558             flock($fp, LOCK_SH);
    559             $fileSize = filesize($localFile);
    560 
    561             $reqBodyLen = 1 + FDFS_PROTO_PKG_LEN_SIZE + FDFS_FILE_EXT_NAME_MAX_LEN + $fileSize;
    562             $reqHeader = FastDFS::packHeader(11, $reqBodyLen);
    563             $reqBody = pack('C', $this->getStorePathIndex()) . FastDFS::packU64($fileSize) . FastDFS::padding($extName, FDFS_FILE_EXT_NAME_MAX_LEN);
    564 
    565             $this->send($reqHeader . $reqBody);
    566 
    567             stream_copy_to_stream($fp, $this->_socket, $fileSize);
    568             flock($fp, LOCK_UN);
    569             fclose($fp);
    570 
    571             $resHeader = $this->read(FDFS_HEADER_LENGTH);
    572             $resInfo = FastDFS::parseHeader($resHeader);
    573 
    574             if ($resInfo['status'] !== 0) {
    575                 return false;
    576             }
    577             $resBody = $resInfo['length'] ? $this->read($resInfo['length']) : '';
    578             $groupName = trim(substr($resBody, 0, FDFS_GROUP_NAME_MAX_LEN));
    579 
    580             $filePath = trim(substr($resBody, FDFS_GROUP_NAME_MAX_LEN));
    581 
    582             if ($metas) {
    583                 $this->setFileMetaData($groupName, $filePath, $metas);
    584             }
    585 
    586             return array(
    587                 'group_name' => $groupName,
    588                 'filename' => $filePath
    589             );
    590         }
    591 
    592         /**
    593          * 
    594          * @param type $fileName
    595          * @param type $groupName
    596          * @param type $masterfile
    597          * @param type $prefix
    598          * @param type $extName
    599          * @param type $metas
    600          * @return boolean
    601          * @throws FastDFSException
    602          */
    603         public function uploadSalveFile($fileName, $groupName, $masterfile, $prefix = '', $extName = '', $metas = array()) {
    604             if (!file_exists($fileName)) {
    605                 throw new FastDFSException("salve file $fileName is not exists");
    606             }
    607 
    608             $pathInfo = pathinfo($fileName);
    609 
    610             $extName = $extName ? $extName : $pathInfo['extension'];
    611             $extLen = strlen($extName);
    612 
    613             if ($extLen > FDFS_FILE_EXT_NAME_MAX_LEN) {
    614                 throw new FastDFSException("salve file ext too long");
    615             }
    616             $fp = fopen($fileName, 'rb');
    617             flock($fp, LOCK_SH);
    618 
    619             $fileSize = filesize($fileName);
    620             $masterFilePathLen = strlen($masterfile);
    621 
    622             $reqBodyLength = 16 + FDFS_FILE_PREFIX_MAX_LEN + FDFS_FILE_EXT_NAME_MAX_LEN + $masterFilePathLen + $fileSize;
    623             $reqHeader = FastDFS::packHeader(FDFS_PROTO_CMD_UPLOAD_SLAVE_FILE, $reqBodyLength);
    624 
    625             $reqBody = pack('x4N', $masterFilePathLen) . FastDFS::packU64($fileSize) . FastDFS::padding($prefix, FDFS_FILE_PREFIX_MAX_LEN);
    626             $reqBody .= FastDFS::padding($extName, FDFS_FILE_EXT_NAME_MAX_LEN) . $masterfile;
    627 
    628             $this->send($reqHeader . $reqBody);
    629 
    630             stream_copy_to_stream($fp, $this->_socket, $fileSize);
    631             flock($fp, LOCK_UN);
    632             fclose($fp);
    633 
    634             $resHeader = $this->read(FDFS_HEADER_LENGTH);
    635             $resInfo = FastDFS::parseHeader($resHeader);
    636 
    637             if ($resInfo['status'] !== 0) {
    638                 return false;
    639             }
    640             $resBody = $resInfo['length'] ? $this->read($resInfo['length']) : '';
    641             $groupName = trim(substr($resBody, 0, FDFS_GROUP_NAME_MAX_LEN));
    642 
    643             $filePath = trim(substr($resBody, FDFS_GROUP_NAME_MAX_LEN));
    644 
    645             if ($metas) {
    646                 $this->setFileMetaData($groupName, $filePath, $metas);
    647             }
    648 
    649             return array(
    650                 'group_name' => $groupName,
    651                 'filename' => $filePath
    652             );
    653         }
    654 
    655         public function deleteFile($groupName, $fileName) {
    656             $reqBodyLen = strlen($fileName) + FDFS_GROUP_NAME_MAX_LEN;
    657             $reqHeader = FastDFS::packHeader(FDFS_PROTO_CMD_DELETE_FILE, $reqBodyLen);
    658             $reqBody = FastDFS::padding($groupName, FDFS_GROUP_NAME_MAX_LEN) . $fileName;
    659 
    660             $this->send($reqHeader . $reqBody);
    661 
    662             $resHeader = $this->read(FDFS_HEADER_LENGTH);
    663             $resInfo = FastDFS::parseHeader($resHeader);
    664 
    665             return !$resInfo['status'];
    666         }
    667 
    668         public function fileExists($groupName, $filePath) {
    669             $meta = $this->getFileMeta($groupName, $filePath);
    670 
    671             return $meta ? true : false;
    672         }
    673 
    674         public function setFileMetaData($groupName, $filePath, array $metaData, $flag = FDFS_OVERWRITE_METADATA) {
    675 
    676             $metaData = FastDFS::packMetaData($metaData);
    677             $metaDataLength = strlen($metaData);
    678             $filePathLength = strlen($filePath);
    679             $flag = $flag === FDFS_OVERWRITE_METADATA ? 'O' : 'M';
    680 
    681             $reqBodyLength = (FDFS_PROTO_PKG_LEN_SIZE * 2) + 1 + $metaDataLength + $filePathLength + FDFS_GROUP_NAME_MAX_LEN;
    682 
    683             $reqHeader = FastDFS::packHeader(FDFS_PROTO_CMD_SET_METADATA, $reqBodyLength);
    684 
    685             $reqBody = FastDFS::packU64($filePathLength) . FastDFS::packU64($metaDataLength);
    686             $reqBody .= $flag . FastDFS::padding($groupName, FDFS_GROUP_NAME_MAX_LEN) . $filePath . $metaData;
    687 
    688             $this->send($reqHeader . $reqBody);
    689 
    690             $resHeader = $this->read(FDFS_HEADER_LENGTH);
    691             $resInfo = FastDFS::parseHeader($resHeader);
    692 
    693             return !$resInfo['status'];
    694         }
    695 
    696         /**
    697          * 取得文件的元信息,如果文件不存在则,返回false,反正是一个关联数组
    698          * 
    699          * @param type $groupName
    700          * @param type $filePath
    701          * @return boolean
    702          */
    703         public function getFileMeta($groupName, $filePath) {
    704             $reqBodyLength = strlen($filePath) + FDFS_GROUP_NAME_MAX_LEN;
    705             $reqHeader = FastDFS::packHeader(FDFS_PROTO_CMD_GET_METADATA, $reqBodyLength);
    706             $reqBody = FastDFS::padding($groupName, FDFS_GROUP_NAME_MAX_LEN) . $filePath;
    707 
    708             $this->send($reqHeader . $reqBody);
    709 
    710             $resHeader = $this->read(FDFS_HEADER_LENGTH);
    711             $resInfo = FastDFS::parseHeader($resHeader);
    712 
    713             if (!!$resInfo['status']) {
    714                 return false;
    715             }
    716 
    717             $resBody = $resInfo['length'] ? $this->read($resInfo['length']) : false;
    718 
    719             return FastDFS::parseMetaData($resBody);
    720         }
    721 
    722         public function close() {
    723             fclose($this->_socket);
    724         }
    725 
    726     }
    727 
    728     class FastDFSException extends Exception {
    729         
    730     }
    731 
    732 }

    上面是api基类:

    下面是调用方式:

     1 /**
     2      * FastDFS上传文件
     3      * @param unknown $fileurl 上传文件
     4      * @param unknown $sizes 附属文件
     5      * @return string
     6      */
     7     private function applydfs($fileurl,$sizes=array()){ 9         $dfs=new FastDFS();
    10         $tracker = $dfs->tracker_get_connection();
    11         $location = "";
    12         if($dfs->active_test($tracker)){
    13             $storaged = $dfs->tracker_query_storage_store("group1",$tracker);
    14             if(!empty($sizes)){
    15                 $count = 0;
    16                 $filename = $dfs->storage_upload_by_filename($fileurl);
    17                 if(isset($filename['group_name'])&&isset($filename['filename'])){
    18                     $location =$filename['group_name']."/".$filename['filename'];
    19                 }
    20                 foreach($sizes as $key=>$val){
    21                     $snapshot_file_info =$dfs->storage_upload_slave_by_filename($val,"group1",$filename['filename'],$key);
    22                     if($snapshot_file_info){
    23                         $count++;
    24                     }
    25                 }
    26                 if($count <> count($sizes)){
    27                     $location = "";
    28                 }
    29             }else{
    30                 $filename = $dfs->storage_upload_by_filename($fileurl);
    31                 if(isset($filename['group_name'])&&isset($filename['filename'])){
    32                     $location =$filename['group_name']."/".$filename['filename'];
    33                 }
    34             }
    35         }
    36         return $location;
    37     }

    在php.ini中配置:

     1 [fastdfs]
     2 ; the base path
     3 fastdfs_client.base_path = /tmp
     4 
     5 ; connect timeout in seconds
     6 ; default value is 30s
     7 fastdfs_client.connect_timeout = 2
     8 
     9 ; network timeout in seconds
    10 ; default value is 30s
    11 fastdfs_client.network_timeout = 60
    12 
    13 ; standard log level as syslog, case insensitive, value list:
    14 ;;; emerg for emergency
    15 ;;; alert
    16 ;;; crit for critical
    17 ;;; error
    18 ;;; warn for warning
    19 ;;; notice
    20 ;;; info
    21 ;;; debug
    22 fastdfs_client.log_level = info
    23 
    24 ; set the log filename, such as /usr/local/fastdfs/logs/fastdfs_client.log
    25 ; empty for output to stderr
    26 fastdfs_client.log_filename = /tmp/fastdfs_client.log
    27 
    28 ; secret key to generate anti-steal token
    29 ; this parameter must be set when http.anti_steal.check_token set to true
    30 ; the length of the secret key should not exceed 128 bytes
    31 fastdfs_client.http.anti_steal_secret_key = 
    32 
    33 ; FastDFS cluster count, default value is 1
    34 fastdfs_client.tracker_group_count = 1
    35 
    36 ; config file of FastDFS cluster ;, based 0
    37 ; must include absolute path, such as fastdfs_client.tracker_group0
    38 ; the config file is same as conf/client.conf
    39 ; 这是你fastdfs的tracker的配置文件的路径,必须使用绝对路径,请换成你自己的路径,附件是我们开发环境和测试环境的client.conf配置
    40 ; 测试环境已经配置过,你自己的开发环境需要配置一下
    41 fastdfs_client.tracker_group0 = "E:/wamp/bin/php/php5.3.13/client.conf"
    42 
    43 ; if use connection pool
    44 ; default value is false
    45 ; since V4.05
    46 fastdfs_client.use_connection_pool = false
    47 
    48 ; connections whose the idle time exceeds this time will be closed
    49 ; unit: second
    50 ; default value is 3600
    51 ; since V4.05
    52 fastdfs_client.connection_pool_max_idle_time = 3600
    53 
    54 ; the base path
    55 fastdfs_client.base_path = /tmp
    56 
    57 ; connect timeout in seconds
    58 ; default value is 30s
    59 fastdfs_client.connect_timeout = 2
    60 
    61 ; network timeout in seconds
    62 ; default value is 30s
    63 fastdfs_client.network_timeout = 60
  • 相关阅读:
    [ES6] 15. Generators -- 2
    [ES6] 14. Generator -- 1. yield & next()
    [ES6] 13. Using the ES6 spread operator ...
    [ES6] 12. Shorthand Properties in ES6
    [ES6] 11. String Templates
    计算机-DB:OLAP(联机分析处理)
    计算机-事务:OLTP(联机事务处理过程)
    软件-数据库-分布式:HBase
    架构-分布式:Hadoop
    计算机-数据仓库:DW/DWH
  • 原文地址:https://www.cnblogs.com/chuanheng/p/3607860.html
Copyright © 2020-2023  润新知