乐趣区

PHP分片上传文件到又拍云对象存储

本文参加又拍云原创技术征文活动

注册 & 实名认证

注册我就不说了,大家自行前往又拍云注册就行。

注册之后需要实名认证,否则无法使用,但由于我已经实名认证了 (当时没截图),网上也很难找到截图,我找了一张很模糊的,不过还是能看清的

认证方式:支付宝芝麻信用有 700 分以上选支付宝芝麻信用认证,没有的话选人工(选人工应该要上传手持身份证的照片)
所属行业:我的是博客,所以我选的是门户网站→博客
网站 / 产品:就填你的网站名称(暂时没有就随便想一个)
网站地址:写你自己搭建的博客域名,没有的话用 csdn,博客园,新浪博客之类的博客地址,再没有的话好像也可以不写
个人名称:写你的姓名(跟身份证要一样)
身份证号码:如实填写,否则无法通过实名认证

创建存储桶

登录后,点击页面右上角的控制台,然后选择云产品→云存储

点击右上角创建一个服务(在这里叫服务,其它云一般叫“存储桶”,英文“bucket”)

按下图填写,其中加速按你的需求,但如果选择全球加速,价格肯定比国内加速贵;操作员如果你之前没有就需要新建并授权,因为调用 api 需要一个账号密码,但这个账号密码并不是你注册又拍云的那个账号密码,而是要在这里新建一个,叫“操作员”

测试操作员账号及默认域名

又拍云的对象存储是支持直接使用 ftp 登录的,这里我使用 filezilla 登录(其它 ftp 工具也类似)。

打开 filezilla,如下图所示,点击左上角的“Site Manager”图标 (也可直接点击菜单FileSite Manager) 进入以下界面,然后按下图填写参数

如果 ftp 能连接上,也能上传文件,说明操作员账号没啥问题,连不上可能是前面忘了授权,可以点击右上角用户名→账户管理

然后点击操作员→管理,查看是否已经给了这个操作员权限,如果没有给,则给它加上权限,然后再试,应该就没问题了

另外,我们还可以在网页上上传文件

创建一个服务后,又拍云默认会给我们一个测试域名,域名结构为”https:// 服务名.test.upcdn.net“,其实也可以在控制台中看到,点击上图中的“配置”→选择 域名管理 选项卡(实际上默认就是该选项卡)→滚动到最底下即可看见送的测试域名。

使用 php 分片上传文件

REST API 文档,其中并行式断点续传和串行式断点续传,就是分片上传了

不过又拍云有写好的 php-sdk,就不用我们自己写这个,直接传参就行。

创建一个文件夹UploadToUpyun,在终端中进入该文件夹,使用 composer 安装 sdk(没有 composer 命令请自行安装,都 0202 年了应该都有吧?)

composer require upyun/sdk

安装完成后,在 UploadToUpyun 文件夹中创建一个index.php,代码如下,注释已经写的非常清楚,填上你的参数即可正常运行,相信没有人不会

<?php
require 'vendor/autoload.php';

use Upyun\Config;
use Upyun\Upyun;

// 你自己创建的服务的名称
$serviceName = 'YOUR_SERVICE_NAME';
// 操作员账号
$operator = 'YOUR_OPERATOR_NAME';
// 操作员密码
$password = 'YOUR_OPERATOR_PASSWORD';
// 要上传到哪个目录下
$directory = '/UploadToUpyun/test/';
// 要上传的文件的绝对路径(请换成你电脑上一个图片文件的路径)
$uploadFilePath = '/Users/bruce/Downloads/Ts8CYigv.png';
// 上传文件名,为什么上传文件名要叫 key 而不叫 filename 呢?因为对象存储其实就是一个 key value
// 的存储结构,你可以认为它就是个 redis,key 就是 redis 键名,value 就是这个 key 对应的内容。$key = basename($uploadFilePath);
// 又拍云默认域名在:云存储→选择一个服务并点配置→配置的第一项默认就是域名管理→滚动到最后就能看到默认域名
$defaultDomain = 'http://' . $serviceName . '.test.upcdn.net';

// 在对象存储中,其实并没有实际意义上的 "目录"(或者叫 "文件夹"),整个文件夹和最后的 key 一起,组成的一个整体也是 key
// 举例:我要把文件存到 "2020/06/02/test.jpg",那么这整个就是一个 key,而内容当然就是 test.jpg 这张图片的二进制数据了。if($directory){
    // 真正使用时,$directory 可能是用户传过来的,在不知道用户是否写了右斜杠的情况下,统一先去掉再添加一个
    $key = rtrim($directory, '/') . '/' . $key;
}

// 使用又拍云 php-sdk 上传文件非常方便,很 new 一个 config 对象
$serviceConfig = new Config($serviceName, $operator, $password);
// 15728640 = 15M,如果文件大于 15M,则使用并行分块上传
if(filesize($uploadFilePath) > 15728640){
    /* uploadType 有两个值
       - BLOCK : 串行分块上传
       - BLOCK_PARALLEL : 并行分块上传
    */
    // 然后可对 new 出的 config 对象的属性继续设置参数(有哪些参数可直接查看 Config 类: vendor/upyun/sdk/src/Upyun/Config.php)
    // 设置 uploadType=BLOCK_PARALLEL 表示使用并行分片上传
    $serviceConfig->uploadType = 'BLOCK_PARALLEL';
}
$client = new Upyun($serviceConfig);
$fp = fopen($uploadFilePath, 'rb');
$retArr = $client->write($key, $fp);
// var_dump($retArr);exit;

/*
 * 返回的 $retArr 的值
 * array(6) {
      'x-upyun-content-length' =>
      string(5) "47028"
      'x-upyun-height' =>
      string(3) "473"
      'x-upyun-content-type' =>
      string(9) "image/png"
      'x-upyun-file-type' =>
      string(3) "PNG"
      'x-upyun-width' =>
      string(3) "839"
      'x-upyun-frames' =>
      string(1) "1"
    }
 */

// 我们随便拿一个元素来判断是否上传成功
if(isset($retArr['x-upyun-content-length'])){
    // 在实际使用中,$key 可能是用户传过来的,在不知道 $key 是否以斜杠开头的情况下,统一先去掉再自己添加一个
    echo $defaultDomain . '/' . ltrim($key);
}else{echo 'Upload failed.';}

好了,到这里就写完了,enjoy!

退出移动版