通过又拍云存储REST API ,为ECStore新增图片存储引擎,从而达到图片数据与主站数据分离。提高网站性能。
  1. 图片存储引擎相关文件添加与修改

一共涉及到ECStore 2个APP 的文件添加和修改。

新增 :app/base/lib/storage/upyunclass.php

<?php

class UpYunException extends Exception {/*{{{*/
public function __construct($message, $code, Exception $previous = null) {
parent::__construct($message, $code); // For PHP 5.2.x
} public function __toString() {
return __CLASS__ . ": [{$this->code}]: {$this->message}\n";
}
}/*}}}*/ class UpYunAuthorizationException extends UpYunException {/*{{{*/
public function __construct($message, $code = 0, Exception $previous = null) {
parent::__construct($message, 401, $previous);
}
}/*}}}*/ class UpYunForbiddenException extends UpYunException {/*{{{*/
public function __construct($message, $code = 0, Exception $previous = null) {
parent::__construct($message, 403, $previous);
}
}/*}}}*/ class UpYunNotFoundException extends UpYunException {/*{{{*/
public function __construct($message, $code = 0, Exception $previous = null) {
parent::__construct($message, 404, $previous);
}
}/*}}}*/ class UpYunNotAcceptableException extends UpYunException {/*{{{*/
public function __construct($message, $code = 0, Exception $previous = null) {
parent::__construct($message, 406, $previous);
}
}/*}}}*/ class UpYunServiceUnavailable extends UpYunException {/*{{{*/
public function __construct($message, $code = 0, Exception $previous = null) {
parent::__construct($message, 503, $previous);
}
}/*}}}*/ class base_storage_upyunclass {
const VERSION = '2.0'; /*{{{*/
const ED_AUTO = 'v0.api.upyun.com';
const ED_TELECOM = 'v1.api.upyun.com';
const ED_CNC = 'v2.api.upyun.com';
const ED_CTT = 'v3.api.upyun.com'; const CONTENT_TYPE = 'Content-Type';
const CONTENT_MD5 = 'Content-MD5';
const CONTENT_SECRET = 'Content-Secret'; // 缩略图
const X_GMKERL_THUMBNAIL = 'x-gmkerl-thumbnail';
const X_GMKERL_TYPE = 'x-gmkerl-type';
const X_GMKERL_VALUE = 'x-gmkerl-value';
const X_GMKERL_QUALITY = 'x­gmkerl-quality';
const X_GMKERL_UNSHARP = 'x­gmkerl-unsharp';
/*}}}*/ private $_bucket_name;
private $_username;
private $_password;
private $_timeout = 30; /**
* @deprecated
*/
private $_content_md5 = NULL; /**
* @deprecated
*/
private $_file_secret = NULL; /**
* @deprecated
*/
private $_file_infos= NULL; protected $endpoint; /**
* 初始化 UpYun 存储接口
* @param $bucketname 空间名称
* @param $username 操作员名称
* @param $password 密码
*
* @return object
*/
public function __construct($bucketname, $username, $password, $endpoint = NULL, $timeout = 30) {/*{{{*/
$this->_bucketname = $bucketname;
$this->_username = $username;
$this->_password = md5($password);
$this->_timeout = $timeout; $this->endpoint = is_null($endpoint) ? self::ED_AUTO : $endpoint;
}/*}}}*/ /**
* 获取当前SDK版本号
*/
public function version() {
return self::VERSION;
} /**
* 创建目录
* @param $path 路径
* @param $auto_mkdir 是否自动创建父级目录,最多10层次
*
* @return void
*/
public function makeDir($path, $auto_mkdir = false) {/*{{{*/
$headers = array('Folder' => 'true');
if ($auto_mkdir) $headers['Mkdir'] = 'true';
return $this->_do_request('PUT', $path, $headers);
}/*}}}*/ /**
* 删除目录和文件
* @param string $path 路径
*
* @return boolean
*/
public function delete($path) {/*{{{*/
return $this->_do_request('DELETE', $path);
}/*}}}*/ /**
* 上传文件
* @param string $path 存储路径
* @param mixed $file 需要上传的文件,可以是文件流或者文件内容
* @param boolean $auto_mkdir 自动创建目录
* @param array $opts 可选参数
*/
public function writeFile($path, $file, $auto_mkdir = False, $opts = NULL) {/*{{{*/
if (is_null($opts)) $opts = array();
if (!is_null($this->_content_md5) || !is_null($this->_file_secret)) {
//if (!is_null($this->_content_md5)) array_push($opts, self::CONTENT_MD5 . ": {$this->_content_md5}");
//if (!is_null($this->_file_secret)) array_push($opts, self::CONTENT_SECRET . ": {$this->_file_secret}");
if (!is_null($this->_content_md5)) $opts[self::CONTENT_MD5] = $this->_content_md5;
if (!is_null($this->_file_secret)) $opts[self::CONTENT_SECRET] = $this->_file_secret;
} // 如果设置了缩略版本或者缩略图类型,则添加默认压缩质量和锐化参数
//if (isset($opts[self::X_GMKERL_THUMBNAIL]) || isset($opts[self::X_GMKERL_TYPE])) {
// if (!isset($opts[self::X_GMKERL_QUALITY])) $opts[self::X_GMKERL_QUALITY] = 95;
// if (!isset($opts[self::X_GMKERL_UNSHARP])) $opts[self::X_GMKERL_UNSHARP] = 'true';
//} if ($auto_mkdir === True) $opts['Mkdir'] = 'true'; $this->_file_infos = $this->_do_request('PUT', $path, $opts, $file); return $this->_file_infos;
}/*}}}*/ /**
* 下载文件
* @param string $path 文件路径
* @param mixed $file_handle
*
* @return mixed
*/
public function readFile($path, $file_handle = NULL) {/*{{{*/
return $this->_do_request('GET', $path, NULL, NULL, $file_handle);
}/*}}}*/ /**
* 获取目录文件列表
*
* @param string $path 查询路径
*
* @return mixed
*/
public function getList($path = '/') {/*{{{*/
$rsp = $this->_do_request('GET', $path); $list = array();
if ($rsp) {
$rsp = explode("\n", $rsp);
foreach($rsp as $item) {
@list($name, $type, $size, $time) = explode("\t", trim($item));
if (!empty($time)) {
$type = $type == 'N' ? 'file' : 'folder';
} $item = array(
'name' => $name,
'type' => $type,
'size' => intval($size),
'time' => intval($time),
);
array_push($list, $item);
}
} return $list;
}/*}}}*/ /**
* 获取目录空间使用情况
*
* @param string $path 目录路径
*
* @return mixed
*/
public function getFolderUsage($path) {/*{{{*/
$rsp = $this->_do_request('GET', $path . '?usage');
return floatval($rsp);
}/*}}}*/ /**
* 获取文件、目录信息
*
* @param string $path 路径
*
* @return mixed
*/
public function getFileInfo($path) {/*{{{*/
$rsp = $this->_do_request('HEAD', $path); return $rsp;
}/*}}}*/ /**
* 连接签名方法
* @param $method 请求方式 {GET, POST, PUT, DELETE}
* return 签名字符串
*/
private function sign($method, $uri, $date, $length){/*{{{*/
//$uri = urlencode($uri);
$sign = "{$method}&{$uri}&{$date}&{$length}&{$this->_password}";
return 'UpYun '.$this->_username.':'.md5($sign);
}/*}}}*/ /**
* HTTP REQUEST 封装
* @param string $method HTTP REQUEST方法,包括PUT、POST、GET、OPTIONS、DELETE
* @param string $path 除Bucketname之外的请求路径,包括get参数
* @param array $headers 请求需要的特殊HTTP HEADERS
* @param array $body 需要POST发送的数据
*
* @return mixed
*/
protected function _do_request($method, $path, $headers = NULL, $body= NULL, $file_handle= NULL) {/*{{{*/
$uri = "/{$this->_bucketname}{$path}";
$ch = curl_init("http://{$this->endpoint}{$uri}"); $_headers = array('Expect:');
if (!is_null($headers) && is_array($headers)){
foreach($headers as $k => $v) {
array_push($_headers, "{$k}: {$v}");
}
} $length = 0;
$date = gmdate('D, d M Y H:i:s \G\M\T'); if (!is_null($body)) {
if(is_resource($body)){
fseek($body, 0, SEEK_END);
$length = ftell($body);
fseek($body, 0); array_push($_headers, "Content-Length: {$length}");
curl_setopt($ch, CURLOPT_INFILE, $body);
curl_setopt($ch, CURLOPT_INFILESIZE, $length);
}
else {
$length = @strlen($body);
array_push($_headers, "Content-Length: {$length}");
curl_setopt($ch, CURLOPT_POSTFIELDS, $body);
}
}
else {
array_push($_headers, "Content-Length: {$length}");
} array_push($_headers, "Authorization: {$this->sign($method, $uri, $date, $length)}");
array_push($_headers, "Date: {$date}"); curl_setopt($ch, CURLOPT_HTTPHEADER, $_headers);
curl_setopt($ch, CURLOPT_TIMEOUT, $this->_timeout);
curl_setopt($ch, CURLOPT_HEADER, 1);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
//curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 0);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method); if ($method == 'PUT' || $method == 'POST') {
curl_setopt($ch, CURLOPT_POST, 1);
}
else {
curl_setopt($ch, CURLOPT_POST, 0);
} if ($method == 'GET' && is_resource($file_handle)) {
curl_setopt($ch, CURLOPT_HEADER, 0);
curl_setopt($ch, CURLOPT_FILE, $file_handle);
} if ($method == 'HEAD') {
curl_setopt($ch, CURLOPT_NOBODY, true);
} $response = curl_exec($ch);
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE); if ($http_code == 0) throw new UpYunException('Connection Failed', $http_code); curl_close($ch); $header_string = '';
$body = ''; if ($method == 'GET' && is_resource($file_handle)) {
$header_string = '';
$body = $response;
}
else {
list($header_string, $body) = explode("\r\n\r\n", $response, 2);
} //var_dump($http_code);
if ($http_code == 200) {
if ($method == 'GET' && is_null($file_handle)) {
return $body;
}
else {
$data = $this->_getHeadersData($header_string);
return count($data) > 0 ? $data : true;
}
//elseif ($method == 'HEAD') {
// //return $this->_get_headers_data(substr($response, 0 , $header_size));
// return $this->_getHeadersData($header_string);
//}
//return True;
}
else {
$message = $this->_getErrorMessage($header_string);
if (is_null($message) && $method == 'GET' && is_resource($file_handle)) {
$message = 'File Not Found';
}
switch($http_code) {
case 401:
throw new UpYunAuthorizationException($message);
break;
case 403:
throw new UpYunForbiddenException($message);
break;
case 404:
throw new UpYunNotFoundException($message);
break;
case 406:
throw new UpYunNotAcceptableException($message);
break;
case 503:
throw new UpYunServiceUnavailable($message);
break;
default:
throw new UpYunException($message, $http_code);
}
}
}/*}}}*/ /**
* 处理HTTP HEADERS中返回的自定义数据
*
* @param string $text header字符串
*
* @return array
*/
private function _getHeadersData($text) {/*{{{*/
$headers = explode("\r\n", $text);
$items = array();
foreach($headers as $header) {
$header = trim($header);
if(strpos($header, 'x-upyun') !== False){
list($k, $v) = explode(':', $header);
$items[trim($k)] = in_array(substr($k,8,5), array('width','heigh','frame')) ? intval($v) : trim($v);
}
}
return $items;
}/*}}}*/ /**
* 获取返回的错误信息
*
* @param string $header_string
*
* @return mixed
*/
private function _getErrorMessage($header_string) {
list($status, $stash) = explode("\r\n", $header_string, 2);
list($v, $code, $message) = explode(" ", $status, 3);
return $message;
} /**
* 删除目录
* @deprecated
* @param $path 路径
*
* @return void
*/
public function rmDir($path) {/*{{{*/
$this->_do_request('DELETE', $path);
}/*}}}*/ /**
* 删除文件
*
* @deprecated
* @param string $path 要删除的文件路径
*
* @return boolean
*/
public function deleteFile($path) {/*{{{*/
$rsp = $this->_do_request('DELETE', $path);
}/*}}}*/ /**
* 获取目录文件列表
* @deprecated
*
* @param string $path 要获取列表的目录
*
* @return array
*/
public function readDir($path) {/*{{{*/
return $this->getList($path);
}/*}}}*/ /**
* 获取空间使用情况
*
* @deprecated 推荐直接使用 getFolderUsage('/')来获取
* @return mixed
*/
public function getBucketUsage() {/*{{{*/
return $this->getFolderUsage('/');
}/*}}}*/ /**
* 获取文件信息
*
* #deprecated
* @param $file 文件路径(包含文件名)
* return array('type'=> file | folder, 'size'=> file size, 'date'=> unix time) 或 null
*/
//public function getFileInfo($file){/*{{{*/
// $result = $this->head($file);
// if(is_null($r))return null;
// return array('type'=> $this->tmp_infos['x-upyun-file-type'], 'size'=> @intval($this->tmp_infos['x-upyun-file-size']), 'date'=> @intval($this->tmp_infos['x-upyun-file-date']));
//}/*}}}*/ /**
* 切换 API 接口的域名
*
* @deprecated
* @param $domain {默然 v0.api.upyun.com 自动识别, v1.api.upyun.com 电信, v2.api.upyun.com 联通, v3.api.upyun.com 移动}
* return null;
*/
public function setApiDomain($domain){/*{{{*/
$this->endpoint = $domain;
}/*}}}*/ /**
* 设置待上传文件的 Content-MD5 值(如又拍云服务端收到的文件MD5值与用户设置的不一致,将回报 406 Not Acceptable 错误)
*
* @deprecated
* @param $str (文件 MD5 校验码)
* return null;
*/
public function setContentMD5($str){/*{{{*/
$this->_content_md5 = $str;
}/*}}}*/ /**
* 设置待上传文件的 访问密钥(注意:仅支持图片空!,设置密钥后,无法根据原文件URL直接访问,需带 URL 后面加上 (缩略图间隔标志符+密钥) 进行访问)
* 如缩略图间隔标志符为 ! ,密钥为 bac,上传文件路径为 /folder/test.jpg ,那么该图片的对外访问地址为: http://空间域名/folder/test.jpg!bac
*
* @deprecated
* @param $str (文件 MD5 校验码)
* return null;
*/
public function setFileSecret($str){/*{{{*/
$this->_file_secret = $str;
}/*}}}*/ /**
* @deprecated
* 获取上传文件后的信息(仅图片空间有返回数据)
* @param $key 信息字段名(x-upyun-width、x-upyun-height、x-upyun-frames、x-upyun-file-type)
* return value or NULL
*/
public function getWritedFileInfo($key){/*{{{*/
if(!isset($this->_file_infos))return NULL;
return $this->_file_infos[$key];
}/*}}}*/
}

新增:app/base/lib/storage/upyunsystem.php

 <?php

 class base_storage_upyunsystem implements base_interface_storager
{
private $_tmpfiles = array(); function __construct()
{
$this->upyun = new base_storage_upyunclass('图片空间名','授权操作员用户名','授权操作员密码');//替换成您自己的upyun信息
$this->UPYUNHOST = array(
'http://img1.yourdomain.com', //替换成您自己的upyun信息,可在upyun后台解析一个多个域名
'http://img2.yourdomain.com',
'http://img3.yourdomain.com'
);
}//End Function public function watermark($file,$mark){
//todo
}
public function resize($file,$w,$h){ if(!$w || !$h)return false;
$opts = array(
'x-gmkerl-type' => 'fix_width_or_height', // 缩略图类型
'x-gmkerl-value' => $w.'x'.$h, // 缩略图大小
'x-gmkerl-quality' => 100, // 缩略图压缩质量
'x-gmkerl-unsharp' => true // 是否进行锐化处理
); $tmp_ident = md5(rand(0,9999).microtime());
$tmp_ident = '/temp/'.substr($tmp_ident,6); $_s = $this->upyun->writeFile( $tmp_ident, file_get_contents($file), true, $opts); if($_s && $_s['x-upyun-width']>0){
$_id = base64_encode($tmp_ident);
$_return = $this->getFile($_id);
$this->remove($_id);
return $_return;
}else{
return false;
} } public function save($file, &$url, $type, $addons, $ext_name="")
{ $filename = '/'.$type.'/'.$this->_get_ident($file,$type,$addons,$url,$path,$ext_name); $_s = $this->upyun->writeFile($filename,file_get_contents($file),true); if($_s && $_s['x-upyun-width']>0){
$url = $this->UPYUNHOST[rand(0,count($this->UPYUNHOST)-1)].$filename;
return base64_encode($filename);
}else{
return false;
}
}//End Function // 生成文件名
public function _get_ident($file,$type,$addons,$url,&$path,$ext_name=""){
$ident = md5(rand(0,9999).microtime());
// 路径
if(isset($addons['path']) && $addons['path']) {
$path = $addons['path'];
} else {
$path = '/'.$ident{0}.$ident{1}.'/'.$ident{2}.$ident{3}.'/'.$ident{4}.$ident{5}.'/'.substr($ident,6);
}
// 文件名
if(isset($addons['name']) && $addons['name']) {
$uri = $addons['name'];
} else {
$uri = substr(md5(($addons?$addons:$file).microtime()),0,6);
}
// 后缀
if($ext_name) {
if(strrpos($uri,".")) $uri = substr($uri,0,strrpos($uri,".")).$ext_name;
else $uri .= $ext_name;
}
return $uri;
} // end function _get_ident public function replace($file, $id)
{
return $this->upyun->writeFile(base64_decode($id), file_get_contents($file) , true);
}//End Function public function remove($id)
{
if($id){
return $this->upyun->deleteFile(base64_decode($id));
}else{
return false;
}
}//End Function public function getFile($id, $type='image')
{
$tmpfile = tempnam(sys_get_temp_dir(), 'upyunsystem');
array_push($this->_tmpfiles, $tmpfile);
$_s = $this->upyun->readFile(base64_decode($id),null);
file_put_contents($tmpfile, $_s);
if($_s){
return $tmpfile;
}else{
return false;
}
}//End Function function __destruct()
{
foreach($this->_tmpfiles AS $tmpfile){
@unlink($tmpfile);
}//todo unlink tmpfiles;
}//End Function }//End Class

注意:此文件内有upyun权限及图片空间域名信息,如直接使用此文件,请做修改!

修改:app/image/lib/clip.php 2处:

修改1:

修改2:

修改:app/image/model/image.php

修改config/config.php ECStore站点配置文件

修改(去除注释)#define('FILE_STORAGER','filesystem'); 为define('FILE_STORAGER','upyunsystem');

修改后实现功能

  1. ECStore所有图片上传功能(在商品相册上传图片、在图片相册上传图片、在图库上传图片,统一会上传到upyun api指定帐户空间,并返回"随机已绑定域名+存储目录+图片后缀"形式的绝对URL
  2. 商品批量重新生成图片将调用upyun API并生成对应尺寸图片绝对URL到对应图片尺寸数据库字段
  3. 图库批量图片尺寸重新生成队列将调用upyun API并生成对应尺寸图片绝对URL到对应图片尺寸数据库字段

最终实现全站图片云端集群存储。

原文链接:http://www.it.toggle.cn/article_detail/b7e80163a2f5b56ee4dfc1bd2dac0ba0.html

ECStore图片云端集群存储实践-又拍云存储的更多相关文章

  1. 转】Neo4j集群安装实践

    原博文出自于: http://blog.fens.me/category/%E6%95%B0%E6%8D%AE%E5%BA%93/page/2/ 感谢! Posted: Oct 29, 2013 Ta ...

  2. PB级数据实时查询,滴滴Elasticsearch多集群架构实践

    PB级数据实时查询,滴滴Elasticsearch多集群架构实践  mp.weixin.qq.com 点击上方"IT牧场",选择"设为星标"技术干货每日送达 点 ...

  3. Nginx+Keepalived高可用集群应用实践

    Nginx+Keepalived高可用集群应用实践 1.Keepalived高可用软件 1.1 Keepalived服务的三个重要功能 1.1.1管理LVS负载均衡软件 早期的LVS软件,需要通过命令 ...

  4. 基于winserver的Apollo配置中心分布式&集群部署实践(正确部署姿势)

    基于winserver的Apollo配置中心分布式&集群部署实践(正确部署姿势)   前言 前几天对Apollo配置中心的demo进行一个部署试用,现公司已决定使用,这两天进行分布式部署的时候 ...

  5. 使用Ceph集群作为Kubernetes的动态分配持久化存储(转)

    使用Docker快速部署Ceph集群 , 然后使用这个Ceph集群作为Kubernetes的动态分配持久化存储. Kubernetes集群要使用Ceph集群需要在每个Kubernetes节点上安装ce ...

  6. Nginx+Memcached+Tomcat集群配置实践(Sticky Session)

    准备工作 创建一个简单的web应用,名为session.其中有两个页面,分别如下所示: 页面login.jsp <%@ page language="java" conten ...

  7. 又拍云叶靖:OpenResty 在又拍云存储中的应用

    2019 年 7 月 6 日,OpenResty 社区联合又拍云,举办 OpenResty × Open Talk 全国巡回沙龙·上海站,又拍云平台开发部负责人叶靖在活动上做了<OpenRest ...

  8. 混合云存储组合拳:基于云存储网关与混合云备份的OSS数据备份方案

    前言 阿里云对象存储(OSS)用户众多.很多用户因为业务或者合规性需求,需要对OSS内的数据做备份,无论是线上备份,还是线下备份.用户可以选择使用OSS的开放API,按照业务需求,做数据的备份,也可以 ...

  9. MongoDB(7):集群部署实践,包含复制集,分片

    注: 刚开始学习MongoDB,写的有点麻烦了,网上教程都是很少的代码就完成了集群的部署, 纯属个人实践,错误之处望指正!有好的建议和资料请联系我QQ:1176479642 集群架构: 2mongos ...

随机推荐

  1. 分布式理论(八)—— Consistent Hash(一致性哈希算法)

    前言 在分布式系统中,常常需要使用缓存,而且通常是集群,访问缓存和添加缓存都需要一个 hash 算法来寻找到合适的 Cache 节点.但,通常不是用取余hash,而是使用我们今天的主角-- 一致性 h ...

  2. .16-浅析webpack源码之编译后流程梳理

    这节把编译打包后的流程梳理一下,然后集中处理compile. 之前忽略了一个点,如下: new NodeEnvironmentPlugin().apply(compiler); // 引入插件加载 i ...

  3. apache 隐藏 index.php

    在根目录下添加文件 .htaccess <IfModule mod_rewrite.c> Options +FollowSymlinks RewriteEngine On RewriteC ...

  4. PowerDesigner Constraint name uniqueness 错误

    使用PowerDesigner生成数据库脚本时报 Constraint name uniqueness 错误: 双击每行错误,发现外键引用的名字有重复的: 惯性去网上找解决办法,找到的主要是两个方法: ...

  5. PCA算法Python实现

    源代码: #-*- coding: UTF-8 -*- from numpy import * import numpy def pca(X,CRate): #矩阵X每行是一个样本 #对样本矩阵进行中 ...

  6. MySQL闪退问题的解决

    刚刚学习了数据库,并且安装了MySQL,正当高兴之余,发现我的MySQL出现了闪退的显现.上网搜了好久的解决方案.最后解决了这个问题,也舒心了. 问题从这里开始: 接着我打开MySQL,寻思能不能用, ...

  7. Spring Boot学习笔记(七)多数据源下的事务管理

    DataBaseConfig中加入事务管理器 DataBaseConfig的详解以及多数据源的配置参见我的上一篇文章 @Configuration @MapperScan(basePackages={ ...

  8. Mysql添加字段.md

    alter table td_user add gender bit DEFAULT 0 COMMENT '性别';

  9. linux系统编程:read,write与lseek的综合应用

    这个实例根据命令行参数进行相应的读学操作: 用法: usage:./io file {r<length>|R<length>|w<string>|s<offs ...

  10. C++ 的那些坑 (Day 1)

    永远的溢出 运算溢出 溢出是一个永恒的话题. int a = 0xf000; int b = 0xff000; long c = a * b; 此时c = -251658240,以为已经使用了long ...