当前位置: 首页 > news >正文

thinkcmf改存储CloudflareR2

先建一个服务类

<?php namespace app\portal\service; class CloudflareR2 { private $accessKey="42098b06dbxxxxxxxx48fad96f4f"; private $secretKey="02d3f18c4ac204d35xxxxxxxxxxxx73bcab3e4c01c982b16fe56c7a7951"; private $accountId="e94bae5daxxxxxx3abbd269e5e"; private $bucketName="suxxxxxoda"; private $region="auto"; public function __construct() { } /** * 获取存储桶 URL */ public function getBucketUrl() { return "{$this->accountId}.r2.cloudflarestorage.com/{$this->bucketName}"; } /** * 上传文件到 Cloudflare R2 */ public function uploadFile($filePath, $objectKey) { if (!file_exists($filePath)) { return ['success' => false, 'error' => 'File not found']; } // 使用同一个时间戳确保所有日期一致 $now = time(); $amzDate = gmdate('Ymd\THis\Z', $now); $dateStamp = gmdate('Ymd', $now); $fileSize = filesize($filePath); $contentType = mime_content_type($filePath); $fileContent = file_get_contents($filePath); // 1. 创建规范请求 $canonicalHeaders = [ 'content-length' => $fileSize, 'content-type' => $contentType, 'host' => "{$this->accountId}.r2.cloudflarestorage.com", 'x-amz-content-sha256' => hash('sha256', $fileContent), 'x-amz-date' => $amzDate ]; // 按字母顺序排序头 ksort($canonicalHeaders); $signedHeaders = implode(';', array_map('strtolower', array_keys($canonicalHeaders))); $canonicalRequest = "PUT\n" . '/' . $this->bucketName . '/' . str_replace('%2F', '/', rawurlencode($objectKey)) . "\n" . "\n" . implode("\n", array_map(function($k, $v) { return strtolower($k) . ':' . $v; }, array_keys($canonicalHeaders), $canonicalHeaders)) . "\n" . "\n" . $signedHeaders . "\n" . $canonicalHeaders['x-amz-content-sha256']; // 2. 创建待签字符串 $algorithm = "AWS4-HMAC-SHA256"; $credentialScope = "{$dateStamp}/{$this->region}/s3/aws4_request"; $stringToSign = "{$algorithm}\n{$amzDate}\n{$credentialScope}\n" . hash('sha256', $canonicalRequest); // 3. 计算签名 $signingKey = $this->getSignatureKey($dateStamp); $signature = hash_hmac('sha256', $stringToSign, $signingKey); // 4. 构建授权头 $authorizationHeader = "{$algorithm} " . "Credential={$this->accessKey}/{$credentialScope}, " . "SignedHeaders={$signedHeaders}, " . "Signature={$signature}"; // 5. 构造请求URL和头 $url = "https://{$this->accountId}.r2.cloudflarestorage.com/{$this->bucketName}/" . rawurlencode($objectKey); $headers = [ "Authorization: {$authorizationHeader}", "Content-Length: {$fileSize}", "Content-Type: {$contentType}", "Host: {$this->accountId}.r2.cloudflarestorage.com", "x-amz-content-sha256: {$canonicalHeaders['x-amz-content-sha256']}", "x-amz-date: {$amzDate}" ]; // 6. 发送请求 $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_CUSTOMREQUEST, "PUT"); curl_setopt($ch, CURLOPT_POSTFIELDS, $fileContent); curl_setopt($ch, CURLOPT_HTTPHEADER, $headers); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $response = curl_exec($ch); $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE); //var_dump($response); die(); curl_close($ch); if ($httpCode === 200) { $url2 = env('CLOUDFLARE_URL') . rawurlencode($objectKey); return ['success' => true, 'url' => $url2]; } else { return [ 'success' => false, 'error' => 'Upload failed', 'http_code' => $httpCode, 'response' => $response, 'debug' => [ 'canonicalRequest' => $canonicalRequest, 'stringToSign' => $stringToSign, 'signingKey' => bin2hex($this->getSignatureKey($dateStamp)), 'signature' => $signature, 'headers' => $headers ] ]; } } /** * 生成签名密钥 */ private function getSignatureKey($dateStamp) { $kSecret = 'AWS4' . $this->secretKey; $kDate = hash_hmac('sha256', $dateStamp, $kSecret, true); $kRegion = hash_hmac('sha256', $this->region, $kDate, true); $kService = hash_hmac('sha256', 's3', $kRegion, true); $kSigning = hash_hmac('sha256', 'aws4_request', $kService, true); return $kSigning; } }

再来个调用的

<?php namespace app\portal\service; use cmf\lib\Storage; class CloudflareR2Storage extends Storage { /** * 上传文件到 Cloudflare R2 * @param string $file 文件路径 * @param string $filePath 本地文件路径 * @param string $fileType 文件类型 * @param array $param 其他参数 * @return array|bool */ public function upload($file, $filePath, $fileType = 'image', $param = null) { $cloudflareR2 = new CloudflareR2(); $objectKey = ltrim($file, '/'); $result = $cloudflareR2->uploadFile($filePath, $objectKey); if ($result['success']) { // 删除本地文件 // @unlink($filePath); return [ 'preview_url' => $result['url'], 'url' => $result['url'], ]; } else { return false; } } /** * 获取图片预览 URL * @param string $file * @param string $style * @return mixed */ public function getPreviewUrl($file, $style = '') { return "https://" . (new CloudflareR2())->getBucketUrl() . "/" . ltrim($file, '/'); } /** * 获取图片 URL * @param string $file * @param string $style * @return mixed */ public function getImageUrl($file, $style = '') { return "https://" . (new CloudflareR2())->getBucketUrl() . "/" . ltrim($file, '/'); } /** * 获取文件下载 URL * @param string $file * @param int $expires * @return mixed|string */ public function getFileDownloadUrl($file, $expires = 3600) { return "https://" . (new CloudflareR2())->getBucketUrl() . "/" . ltrim($file, '/'); } }

然后找到cmf的扩展,找到对应的上传方法

在他上传的方法前面插入个自己的上传判断

// 检查是否配置了 Cloudflare R2 存储 $storage = cmf_get_option('storage'); if ( env('CLOUDFLARE_STATUS') == 1) { // 使用 Cloudflare R2 存储 $cloudflareR2Storage = new \app\portal\service\CloudflareR2Storage(); $result = $cloudflareR2Storage->upload($arrInfo["file_path"], $uploadPath . $arrInfo["file_path"], $fileType); if ($result) { // 删除本地文件 // @unlink($uploadPath . $arrInfo["file_path"]); return array_merge([ 'filepath' => $arrInfo["file_path"], "name" => $arrInfo["filename"], 'id' => $strId, 'preview_url' => $result['preview_url'], 'url' => $result['url'], ], $result); } }

我是直接写在配置文件读取的,要不要写数据库看自己了.

这样就能实现上传到Cloundflare,接着还要处理回显的问题.找到这函数

,改写他

function cmf_get_image_url($file, $style = 'watermark') { if (empty($file)) { return ''; } if (strpos($file, "http") === 0) { return $file; } else { // 获取云存储配置 $CLOUDFLARE_STATUS = env("CLOUDFLARE_STATUS"); $cloudStorageUrl = env("CLOUDFLARE_URL"); // 如果配置了云存储URL,则直接使用 if ($CLOUDFLARE_STATUS == 1) { // 处理路径 $file = ltrim(str_replace('\\', '/', $file), '/'); //如果不是thems开头的需要加上upload前缀 // if (!preg_match('/^themes/', $file) && !preg_match('/^static/', $file)) { // $file = 'upload/' . $file; // } return rtrim($cloudStorageUrl, '/') . '/' . $file; } else { // 使用默认的存储处理 $storage = Storage::instance(); return $storage->getImageUrl($file, $style); } } }

具体匹配规则看自己实际目录结构了.

这样就完成了上传到Cloudflare.

http://www.jsqmd.com/news/83483/

相关文章:

  • 告别“创意枯竭周期”:华为云Flexus AI智能体如何重构传统企业营销内容生产力
  • springboot共享自习室管理系统_2bm6j0gn-
  • 柠檬 软件测试之python全栈自动化测试工程师第25期
  • 从“我被拒了”到“我学到了”——软件测试面试的精准复盘法则
  • System Informer:你的Windows系统管家,3大核心功能深度解析
  • 如何与猎头高效沟通,获得心仪的SDET岗位推荐?
  • 车载功能测试都要做什么?总结来了~
  • 基于Python的毕业生去向反馈调查平台的设计与实现(源码+lw+部署文档+讲解等)
  • 如何快速解锁RouterOS无限权限:MikroTikPatch的终极指南
  • Optic开源项目:API管理与测试的终极解决方案
  • Sniffnet容器化部署终极指南:3步搞定网络流量监控
  • springboot大学生社团管理系统_z48oy3bd-
  • 测试开发面试题:单例的设计模式和应用场景
  • 快速获取Notion安卓版:2024最新下载安装全攻略
  • 终极Go接口模拟测试:用mockery重新定义高效测试策略
  • springboot家政服务管理系统的设计与实现_z7z041x0-
  • 探索城市脉搏:解密共享单车数据背后的故事
  • 掌握基础SEO知识,助力网站流量与排名提升
  • 云南民宿联盟:破解诗与远方的生存密码
  • CppSharp项目详解:C/C++到.NET的自动化绑定工具指南
  • Cerebro插件开发终极指南:从入门到精通的完整教程
  • 嵌入式软件开发,测试、找bug技巧
  • Percona Grafana仪表板:一站式数据库监控解决方案
  • 基于STM32的电子钟万年历闹钟OLED显示proteus仿真+实物设计(带温湿度显示)
  • 3步掌握JSMpeg播放器数据监控与性能优化实战指南
  • EcoPaste终极指南:3分钟掌握跨平台剪贴板管理
  • VoxCPM语音合成技术深度解析:从技术突破到商业应用
  • vue基于Spring Boot在线C语言教学系统的设计与实现_jq2o82ge-java毕业设计
  • 2025大模型效率革命:Gemma 3 12B实现高性能与低门槛部署新范式
  • 打包后页面出现空白问题