WebUploadToOss.php 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. <?php
  2. declare(strict_types=1);
  3. namespace web2oss;
  4. use think\facade\Cache;
  5. class WebUploadToOss
  6. {
  7. public static function getPolicy($config)
  8. {
  9. // var_dump($config);
  10. // 从缓存中获取
  11. $catch_name = md5($_SERVER['PATH_INFO']);
  12. $response = Cache::get('ALIYUN_OSS_POLICY_'. $catch_name);
  13. if ($response) {
  14. return array('success' => true, 'message' => 'ok', 'data' => $response);
  15. }
  16. // 请求阿里云OSS
  17. if (!$config['id'] || !$config['key'] || !$config['host'] || !$config['dir']) {
  18. return array('success' => false, 'message' => 'failed,missing required argument!', 'data' => '');
  19. }
  20. $id = $config['id']; // AccessKeyId。
  21. $key = $config['key']; // AccessKeySecret。
  22. $host = $config['host']; // $host的格式为 bucketname.endpoint
  23. $callbackUrl = $config['callbackUrl'] ?: ''; // $callbackUrl为上传回调服务器的URL
  24. $callbackBody = isset($config['callbackBody']) ? $config['callbackBody'] : 'path=' . $host . '/${object}&size=${size}&mimeType=${mimeType}&height=${imageInfo.height}&width=${imageInfo.width}';
  25. $callbackBodyType = isset($config['callbackBodyType']) ? $config['callbackBodyType'] : "application/x-www-form-urlencoded";
  26. $dir = $config['dir']; // 用户上传文件时指定的前缀(目录)
  27. $callback_param = array(
  28. 'callbackUrl' => $callbackUrl,
  29. 'callbackBody' => $callbackBody,
  30. 'callbackBodyType' => $callbackBodyType
  31. );
  32. $callback_string = json_encode($callback_param);
  33. $base64_callback_body = base64_encode($callback_string);
  34. $now = time();
  35. $expire = isset($config['expire']) && $config['expire'] > 30 ? (int)$config['expire'] : 60; //设置该policy超时时间(单位:秒). 即这个policy过了这个有效时间,将不能访问。
  36. $end = $now + $expire;
  37. $expiration = self::gmtIso8601($end);
  38. //最大文件大小.用户可以自己设置
  39. $condition = array(0 => 'content-length-range', 1 => 0, 2 => isset($config['fileSize']) ? $config['fileSize'] : 83886080);
  40. $conditions[] = $condition;
  41. // 表示用户上传的数据,必须是以$dir开始,不然上传会失败,这一步不是必须项,只是为了安全起见,防止用户通过policy上传到别人的目录。
  42. $start = array(0 => 'starts-with', 1 => '$key', 2 => $dir);
  43. $conditions[] = $start;
  44. $arr = array('expiration' => $expiration, 'conditions' => $conditions);
  45. $policy = json_encode($arr);
  46. $base64_policy = base64_encode($policy);
  47. $string_to_sign = $base64_policy;
  48. $signature = base64_encode(hash_hmac('sha1', $string_to_sign, $key, true));
  49. $response = array();
  50. $response['accessid'] = $id;
  51. $response['host'] = $host;
  52. $response['policy'] = $base64_policy;
  53. $response['signature'] = $signature;
  54. $response['expire'] = $end;
  55. $response['callback'] = $base64_callback_body;
  56. $response['dir'] = $dir; // 这个参数是设置用户上传文件时指定的前缀。
  57. // 设置缓存
  58. Cache::set('ALIYUN_OSS_POLICY_'. $catch_name, $response, 59);
  59. return array('success' => true, 'message' => 'ok', 'data' => $response);
  60. }
  61. public static function gmtIso8601($time)
  62. {
  63. $dtStr = date("c", $time);
  64. //$mydatetime = new DateTime($dtStr);
  65. //$expiration = $mydatetime->format(DateTime::ISO8601);
  66. $pos = strpos($dtStr, '+');
  67. $expiration = substr($dtStr, 0, $pos);
  68. return $expiration . "Z";
  69. }
  70. public static function callback()
  71. {
  72. // 1.获取OSS的签名header和公钥url header
  73. $authorizationBase64 = "";
  74. $pubKeyUrlBase64 = "";
  75. /*
  76. * 注意:如果要使用HTTP_AUTHORIZATION头,你需要先在apache或者nginx中设置rewrite,以apache为例,修改
  77. * 配置文件/etc/httpd/conf/httpd.conf(以你的apache安装路径为准),在DirectoryIndex index.php这行下面增加以下两行
  78. RewriteEngine On
  79. RewriteRule .* - [env=HTTP_AUTHORIZATION:%{HTTP:Authorization},last]
  80. * */
  81. if (isset($_SERVER['HTTP_AUTHORIZATION'])) {
  82. $authorizationBase64 = $_SERVER['HTTP_AUTHORIZATION'];
  83. }
  84. if (isset($_SERVER['HTTP_X_OSS_PUB_KEY_URL'])) {
  85. $pubKeyUrlBase64 = $_SERVER['HTTP_X_OSS_PUB_KEY_URL'];
  86. }
  87. if ($authorizationBase64 == '' || $pubKeyUrlBase64 == '') {
  88. // header("http/1.1 403 Forbidden");
  89. // exit();
  90. return array('success' => false, 'message' => 'Forbidden,auth failed');
  91. }
  92. // 2.获取OSS的签名
  93. $authorization = base64_decode($authorizationBase64);
  94. // 3.获取公钥
  95. $pubKeyUrl = base64_decode($pubKeyUrlBase64);
  96. $ch = curl_init();
  97. curl_setopt($ch, CURLOPT_URL, $pubKeyUrl);
  98. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
  99. curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
  100. $pubKey = curl_exec($ch);
  101. if ($pubKey == "") {
  102. //header("http/1.1 403 Forbidden");
  103. //exit();
  104. return array('success' => false, 'message' => 'Forbidden,pubKey is empty');
  105. }
  106. // 4.获取回调body
  107. $body = file_get_contents('php://input');
  108. // 5.拼接待签名字符串
  109. $authStr = '';
  110. $path = $_SERVER['REQUEST_URI'];
  111. $pos = strpos($path, '?');
  112. if ($pos === false) {
  113. $authStr = urldecode($path) . "\n" . $body;
  114. } else {
  115. $authStr = urldecode(substr($path, 0, $pos)) . substr($path, $pos, strlen($path) - $pos) . "\n" . $body;
  116. }
  117. // 6.验证签名
  118. $ok = openssl_verify($authStr, $authorization, $pubKey, OPENSSL_ALGO_MD5);
  119. if ($ok == 1) {
  120. $file_path = array('path' => $_POST['path']);
  121. return array('success' => true, 'message' => 'verify success', 'data' => $file_path);
  122. } else {
  123. // header("http/1.1 403 Forbidden");
  124. // exit();
  125. return array('success' => false, 'message' => 'verify failed');
  126. }
  127. }
  128. }