hyperf| hyperf/guzzle 下载文件问题详解

本文涉及的产品
对象存储 OSS,20GB 3个月
对象存储 OSS,恶意文件检测 1000次 1年
对象存储 OSS,内容安全 1000次 1年
简介: 使用 `hyperf/guzzle` 遇到下载文件的问题, 过程虽波折, 却颇有意思, 分享给大家.

使用 hyperf/guzzle 遇到下载文件的问题, 过程虽波折, 却颇有意思, 分享给大家.

业务上有一个下载文件的任务, 太简单啦

业务上要下载一个 oss 文件, 假设地址为 oss_url. 先撸一遍 guzzle 的文档

下载需要在 request option 中设置 sink 参数: http://docs.guzzlephp.org/en/stable/request-options.html#sink
$oss_url = 'oss_url';
$file_path = 'xxx';
$client = new \GuzzleHttp\Client([
    'verify' => false,
    'decode_content' => false,
]);
$client->get($oss_url, [
    'sink' => $file_path,
]);

换成 hyperf/guzzle 来写:

// 使用 clientFactory 获取到协程版 client 即可
$container = ApplicationContext::getContainer();
$clientFactory = $container->get(ClientFactory::class);
$client = $clientFactory->create([
    'verify' => false,
    'decode_content' => false,
]);

开心的执行, 剧情按照预期的方向发展~

并没有!!! 下载没反应!!!

下载地址的问题?

使用 curl/wget 等命令行工具验证

wget oss_url
curl -O oss_url

下载地址没问题

文件有 300M, 会不会是超时了?

$client = $clientFactory->create([
    'timeout' => 600, // 超时设置为 10 分钟
    'verify' => false,
    'decode_content' => false,
]);

噗, 还是一样没效果?

翻 hyperf 文档

hyperf/guzzle: https://doc.hyperf.io/#/zh/guzzle

哦, 原来使用 swoole 配置要这样:

<?php
use GuzzleHttp\Client;
use Hyperf\Guzzle\CoroutineHandler;
use GuzzleHttp\HandlerStack;

$client = new Client([
    'base_uri' => 'http://127.0.0.1:8080',
    'handler' => HandlerStack::create(new CoroutineHandler()),
    'timeout' => 5,
    'swoole' => [ // 看这里看这里
        'timeout' => 10,
        'socket_buffer_size' => 1024 * 1024 * 2,
    ],
]);

$response = $client->get('/');

这次应该行了吧 !?

噗, 还是一样没效果.

直接用 guzzle 行不行

$oss_url = 'oss_url';
$file_path = 'xxx';
$client = new \GuzzleHttp\Client([
    'verify' => false,
    'decode_content' => false,
]);
$client->get($oss_url, [
    'sink' => $file_path,
]);

终于看了需要下载的问题. 是时候刷锅一波给 hyperf, 什么渣渣框架, 文件下载居然都不支持.

我们看看锅到底在哪

老规矩, 看源码, 既然是使用 \Hyperf\Guzzle\ClientFactory->create() 新建的, 看看源码涨啥样:

public function create(array $options = []): Client
{
    $stack = null;
    if (Coroutine::getCid() > 0) {
        $stack = HandlerStack::create(new CoroutineHandler());
    }

    $config = array_replace(['handler' => $stack], $options);

    if (method_exists($this->container, 'make')) {
        // Create by DI for AOP.
        return $this->container->make(Client::class, ['config' => $config]);
    }
    return new Client($config);
}

协程环境下使用的 new CoroutineHandler, 来看看庐山真面目(文件有点长, 不要轻言放弃):

// \Hyperf\Guzzle\CoroutineHandler
// 关键在这句, 这里其实是 \Swoole\Coroutine\Http\Client
$client = new Client($host, $port, $ssl);

原来这里用的 \Swoole\Coroutine\Http\Client. 这时候我灵机一动, 会不会是?

来看 swoole 的文档

下载方法在这里: https://wiki.swoole.com/wiki/page/938.html
$host = 'www.swoole.com';
$cli = new \Swoole\Coroutine\Http\Client($host, 443, true);
$cli->set(['timeout' => -1]);
$cli->setHeaders([
    'Host' => $host,
    "User-Agent" => 'Chrome/49.0.2587.3',
    'Accept' => '*',
    'Accept-Encoding' => 'gzip'
]);
$cli->download('/static/files/swoole-logo.svg', __DIR__ . '/logo.svg');

这 api 和 guzzle 完全不一样呀 !!! 坑爹呢这是, 用 swoole + hyperf, 连个文件下载都搞不定 ?!

swoole + hyperf 表示我这么强大, 你居然不会用 !

swoole v4.4.6 的版本更新中, 就增加了对 curl hook 的支持, 添加 SWOOLE_HOOK_FLAGS 即可, hyperf v1.1.0 版本中已经提供了支持:

// bin/hyperf.php:11
! defined('SWOOLE_HOOK_FLAGS') && define('SWOOLE_HOOK_FLAGS', SWOOLE_HOOK_ALL);

开启后, swoole 就会无缝将 curl hook 为协程版.

无形加速, 最为致命 !

可是 hyperf/guzzle 默认还是使用的 \Swoole\Coroutine\Http\Client 怎么办? 这就太简单了:

  • 如果你喜欢静态类风格:
<?php

namespace App\Util;

use GuzzleHttp\Client;

class Guzzle
{
    /**
     * @param array $config
     * @return Client
     */
    public static function create(array $config)
    {
        return make(Client::class, ['config' => $config]);
    }
}
  • 如果你喜欢 ClientFactory 风格
<?php

namespace App\Util;

use GuzzleHttp\Client;

class ClientFactory
{
    /**
     * @param array $config
     * @return Client
     */
    public function create(array $config)
    {
        return make(Client::class, ['config' => $config]);
    }
}

写在最后

没事不要乱甩锅, 多问问问题, 多找找原因, 收获就在这不经意间

相关实践学习
借助OSS搭建在线教育视频课程分享网站
本教程介绍如何基于云服务器ECS和对象存储OSS,搭建一个在线教育视频课程分享网站。
目录
相关文章
|
4月前
|
小程序 Shell Linux
workman(二)thinkphp5.0安装websocket插件workerman
首先说明一下我使用的PHP框架是thinkphp5.0。 当然,workerman这个插件不是只有thinkphp5.0可以使用。 具体的安装方法,thinkphp5.0的官方手册中是给出了明确的说明 请移步《thinkphp5.0官方手册》
76 0
|
Java API PHP
(转载)为什么不推荐使用swoole和hyperf官方框架
(转载)为什么不推荐使用swoole和hyperf官方框架
3083 0
|
Prometheus 监控 Cloud Native
hyperf| hyperf/metric 上手指南
这期又开始聊微服务的基础设施之一: 实时监控. 更准确的说法, 基于 prometheus 的实时监控. 关于技术选型这里就不多啰嗦啦, 很多时候「从众」或者「用脚投票」往往是最有效的
618 0
|
4月前
|
Dart 前端开发 JavaScript
swoole(一)laravel8配置使用swoole(1)
然后在服务器使用命令行执行 复制代码
99 0
|
4月前
|
前端开发 小程序 JavaScript
swoole(一)laravel8配置使用swoole(3)
Laravel-swoole的优势就是将swoole插件集成至laravel-s插件中,当前laravel-s插件启动的时候,同时持久化的启动了swoole,不需要我们再服务器端再启动一次swoole。
38 0
|
4月前
|
前端开发 应用服务中间件 网络安全
swoole(一)laravel8配置使用swoole(2)
将上边我使用xxx代替的部分改成你自己的信息。 特别说明一下:我这里使用了ssl证书,如果您没有相关的配置,把https那部分换掉就好。 具体的配置含义,请参考官方文档:
49 0
|
PHP
laravel 上传文件到七牛云 zgldh/qiniu-laravel-storage
laravel 上传文件到七牛云 zgldh/qiniu-laravel-storage
232 0
|
PHP 容器
hyperf| hyperf 源码解读 2: start
上篇我们跟着 `php bin/hyperf.php` 命令, 看到了框架的核心 `container`, 这篇我们跟着 `php bin/hyperf.php start`, 来会一会强大到爆炸的 `swoole`
596 0
|
SEO
easyswoole项目示例
easyswoole项目示例
90 0
|
JSON 前端开发 JavaScript
Flask使用bootstrap为HttpServer添加上传文件功能
今天朋友说,之前写的文章Flask开发vip版HttpServer,具备了httpserver的下载功能,比python原生的http服务好用多了,但能否再此基础上添加一个文件上传的功能呢?必须可以啊,正好复习下bootstrap和jQuery的相关知识。
146 0