123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- <?php
- declare(strict_types=1);
- namespace web2oss;
- use think\facade\Cache;
- class WebUploadToOss
- {
- public static function getPolicy($config)
- {
- // var_dump($config);
- // 从缓存中获取
- $catch_name = md5($_SERVER['PATH_INFO']);
- $response = Cache::get('ALIYUN_OSS_POLICY_'. $catch_name);
- if ($response) {
- return array('success' => true, 'message' => 'ok', 'data' => $response);
- }
- // 请求阿里云OSS
- if (!$config['id'] || !$config['key'] || !$config['host'] || !$config['dir']) {
- return array('success' => false, 'message' => 'failed,missing required argument!', 'data' => '');
- }
- $id = $config['id']; // AccessKeyId。
- $key = $config['key']; // AccessKeySecret。
- $host = $config['host']; // $host的格式为 bucketname.endpoint
- $callbackUrl = $config['callbackUrl'] ?: ''; // $callbackUrl为上传回调服务器的URL
- $callbackBody = isset($config['callbackBody']) ? $config['callbackBody'] : 'path=' . $host . '/${object}&size=${size}&mimeType=${mimeType}&height=${imageInfo.height}&width=${imageInfo.width}';
- $callbackBodyType = isset($config['callbackBodyType']) ? $config['callbackBodyType'] : "application/x-www-form-urlencoded";
- $dir = $config['dir']; // 用户上传文件时指定的前缀(目录)
- $callback_param = array(
- 'callbackUrl' => $callbackUrl,
- 'callbackBody' => $callbackBody,
- 'callbackBodyType' => $callbackBodyType
- );
- $callback_string = json_encode($callback_param);
- $base64_callback_body = base64_encode($callback_string);
- $now = time();
- $expire = isset($config['expire']) && $config['expire'] > 30 ? (int)$config['expire'] : 60; //设置该policy超时时间(单位:秒). 即这个policy过了这个有效时间,将不能访问。
- $end = $now + $expire;
- $expiration = self::gmtIso8601($end);
- //最大文件大小.用户可以自己设置
- $condition = array(0 => 'content-length-range', 1 => 0, 2 => isset($config['fileSize']) ? $config['fileSize'] : 83886080);
- $conditions[] = $condition;
- // 表示用户上传的数据,必须是以$dir开始,不然上传会失败,这一步不是必须项,只是为了安全起见,防止用户通过policy上传到别人的目录。
- $start = array(0 => 'starts-with', 1 => '$key', 2 => $dir);
- $conditions[] = $start;
- $arr = array('expiration' => $expiration, 'conditions' => $conditions);
- $policy = json_encode($arr);
- $base64_policy = base64_encode($policy);
- $string_to_sign = $base64_policy;
- $signature = base64_encode(hash_hmac('sha1', $string_to_sign, $key, true));
- $response = array();
- $response['accessid'] = $id;
- $response['host'] = $host;
- $response['policy'] = $base64_policy;
- $response['signature'] = $signature;
- $response['expire'] = $end;
- $response['callback'] = $base64_callback_body;
- $response['dir'] = $dir; // 这个参数是设置用户上传文件时指定的前缀。
- // 设置缓存
- Cache::set('ALIYUN_OSS_POLICY_'. $catch_name, $response, 59);
- return array('success' => true, 'message' => 'ok', 'data' => $response);
- }
- public static function gmtIso8601($time)
- {
- $dtStr = date("c", $time);
- //$mydatetime = new DateTime($dtStr);
- //$expiration = $mydatetime->format(DateTime::ISO8601);
- $pos = strpos($dtStr, '+');
- $expiration = substr($dtStr, 0, $pos);
- return $expiration . "Z";
- }
- public static function callback()
- {
- // 1.获取OSS的签名header和公钥url header
- $authorizationBase64 = "";
- $pubKeyUrlBase64 = "";
- /*
- * 注意:如果要使用HTTP_AUTHORIZATION头,你需要先在apache或者nginx中设置rewrite,以apache为例,修改
- * 配置文件/etc/httpd/conf/httpd.conf(以你的apache安装路径为准),在DirectoryIndex index.php这行下面增加以下两行
- RewriteEngine On
- RewriteRule .* - [env=HTTP_AUTHORIZATION:%{HTTP:Authorization},last]
- * */
- if (isset($_SERVER['HTTP_AUTHORIZATION'])) {
- $authorizationBase64 = $_SERVER['HTTP_AUTHORIZATION'];
- }
- if (isset($_SERVER['HTTP_X_OSS_PUB_KEY_URL'])) {
- $pubKeyUrlBase64 = $_SERVER['HTTP_X_OSS_PUB_KEY_URL'];
- }
- if ($authorizationBase64 == '' || $pubKeyUrlBase64 == '') {
- // header("http/1.1 403 Forbidden");
- // exit();
- return array('success' => false, 'message' => 'Forbidden,auth failed');
- }
- // 2.获取OSS的签名
- $authorization = base64_decode($authorizationBase64);
- // 3.获取公钥
- $pubKeyUrl = base64_decode($pubKeyUrlBase64);
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_URL, $pubKeyUrl);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
- curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
- $pubKey = curl_exec($ch);
- if ($pubKey == "") {
- //header("http/1.1 403 Forbidden");
- //exit();
- return array('success' => false, 'message' => 'Forbidden,pubKey is empty');
- }
- // 4.获取回调body
- $body = file_get_contents('php://input');
- // 5.拼接待签名字符串
- $authStr = '';
- $path = $_SERVER['REQUEST_URI'];
- $pos = strpos($path, '?');
- if ($pos === false) {
- $authStr = urldecode($path) . "\n" . $body;
- } else {
- $authStr = urldecode(substr($path, 0, $pos)) . substr($path, $pos, strlen($path) - $pos) . "\n" . $body;
- }
- // 6.验证签名
- $ok = openssl_verify($authStr, $authorization, $pubKey, OPENSSL_ALGO_MD5);
- if ($ok == 1) {
- $file_path = array('path' => $_POST['path']);
- return array('success' => true, 'message' => 'verify success', 'data' => $file_path);
- } else {
- // header("http/1.1 403 Forbidden");
- // exit();
- return array('success' => false, 'message' => 'verify failed');
- }
- }
- }
|