UeditorController.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. <?php
  2. // +—————————————————————————————————————————————————————————————————————
  3. // | Created by Yunbao
  4. // +—————————————————————————————————————————————————————————————————————
  5. // | Copyright (c) 2013~2022 http://www.yunbaokj.com All rights reserved.
  6. // +—————————————————————————————————————————————————————————————————————
  7. // | Author: https://gitee.com/yunbaokeji
  8. // +—————————————————————————————————————————————————————————————————————
  9. // | Date: 2022-02-17
  10. // +—————————————————————————————————————————————————————————————————————
  11. namespace app\user\controller;
  12. use cmf\controller\HomeBaseController;
  13. use cmf\lib\Upload;
  14. use think\exception\HttpResponseException;
  15. use think\Response;
  16. /**
  17. * 百度编辑器文件上传处理控制器
  18. * Class Ueditor
  19. * @package app\asset\controller
  20. */
  21. class UeditorController extends HomeBaseController
  22. {
  23. private $stateMap = [ //上传状态映射表,国际化用户需考虑此处数据的国际化
  24. "SUCCESS", //上传成功标记,在UEditor中内不可改变,否则flash判断会出错
  25. "文件大小超出 upload_max_filesize 限制",
  26. "文件大小超出 MAX_FILE_SIZE 限制",
  27. "文件未被完整上传",
  28. "没有文件被上传",
  29. "上传文件为空",
  30. "ERROR_TMP_FILE" => "临时文件错误",
  31. "ERROR_TMP_FILE_NOT_FOUND" => "找不到临时文件",
  32. "ERROR_SIZE_EXCEED" => "文件大小超出网站限制",
  33. "ERROR_TYPE_NOT_ALLOWED" => "文件类型不允许",
  34. "ERROR_CREATE_DIR" => "目录创建失败",
  35. "ERROR_DIR_NOT_WRITEABLE" => "目录没有写权限",
  36. "ERROR_FILE_MOVE" => "文件保存时出错",
  37. "ERROR_FILE_NOT_FOUND" => "找不到上传文件",
  38. "ERROR_WRITE_CONTENT" => "写入文件内容错误",
  39. "ERROR_UNKNOWN" => "未知错误",
  40. "ERROR_DEAD_LINK" => "链接不可用",
  41. "ERROR_HTTP_LINK" => "链接不是http链接",
  42. "ERROR_HTTP_CONTENTTYPE" => "链接contentType不正确"
  43. ];
  44. /**
  45. * 初始化
  46. */
  47. public function initialize()
  48. {
  49. $adminId = cmf_get_current_admin_id();
  50. $userId = cmf_get_current_user_id();
  51. if (empty($adminId) && empty($userId)) {
  52. $this->error("非法上传!");
  53. }
  54. }
  55. /**
  56. * 处理上传处理
  57. */
  58. public function upload()
  59. {
  60. // error_reporting(E_ERROR);
  61. // header("Content-Type: text/html; charset=utf-8");
  62. $action = $this->request->param('action');
  63. switch ($action) {
  64. case 'config':
  65. $result = $this->ueditorConfig();
  66. break;
  67. /* 上传图片 */
  68. case 'uploadimage':
  69. $result = $this->ueditorUpload("image");
  70. break;
  71. /* 上传涂鸦 */
  72. case 'uploadscrawl':
  73. $result = $this->ueditorUpload("image");
  74. break;
  75. /* 上传视频 */
  76. case 'uploadvideo':
  77. $result = $this->ueditorUpload("video");
  78. break;
  79. /* 上传文件 */
  80. case 'uploadfile':
  81. $result = $this->ueditorUpload("file");
  82. break;
  83. /* 列出图片 */
  84. case 'listimage':
  85. $result = "";
  86. break;
  87. /* 列出文件 */
  88. case 'listfile':
  89. $result = "";
  90. break;
  91. /* 抓取远程文件 */
  92. case 'catchimage':
  93. $result = $this->_get_remote_image();
  94. break;
  95. default:
  96. $result = json_encode(['state' => '请求地址出错']);
  97. break;
  98. }
  99. /* 输出结果 */
  100. if (isset($_GET["callback"]) && false) {//TODO 跨域上传
  101. if (preg_match("/^[\w_]+$/", $_GET["callback"])) {
  102. echo htmlspecialchars($_GET["callback"]) . '(' . $result . ')';
  103. } else {
  104. echo json_encode([
  105. 'state' => 'callback参数不合法'
  106. ]);
  107. }
  108. } else {
  109. $response = Response::create(json_decode($result,true),'json');
  110. throw new HttpResponseException($response);
  111. }
  112. }
  113. /**
  114. * 获取远程图片
  115. */
  116. private function _get_remote_image()
  117. {
  118. $source = $this->request->param('source');
  119. $item = [
  120. "state" => "",
  121. "url" => "",
  122. "size" => "",
  123. "title" => "",
  124. "original" => "",
  125. "source" => ""
  126. ];
  127. $date = date("Ymd");
  128. $uploadSetting = cmf_get_upload_setting();
  129. $uploadMaxFileSize = $uploadSetting["image"]['upload_max_filesize'];
  130. $uploadMaxFileSize = empty($uploadMaxFileSize) ? 2048 : $uploadMaxFileSize;//默认2M
  131. $allowedExts = explode(',', $uploadSetting["image"]["extensions"]);
  132. $strSavePath = ROOT_PATH . 'public' . DIRECTORY_SEPARATOR . "ueditor" . DIRECTORY_SEPARATOR . $date . DIRECTORY_SEPARATOR;
  133. //远程抓取图片配置
  134. $config = [
  135. "savePath" => $strSavePath, //保存路径
  136. "allowFiles" => $allowedExts,// [".gif", ".png", ".jpg", ".jpeg", ".bmp"], //文件允许格式
  137. "maxSize" => $uploadMaxFileSize //文件大小限制,单位KB
  138. ];
  139. $storage_setting = cmf_get_cmf_settings('storage');
  140. $qiniu_domain = $storage_setting['Qiniu']['domain'];
  141. $no_need_domains = [$qiniu_domain];
  142. $list = [];
  143. foreach ($source as $imgUrl) {
  144. $host = str_replace(['http://', 'https://'], '', $imgUrl);
  145. $host = explode('/', $host);
  146. $host = $host[0];
  147. if (in_array($host, $no_need_domains)) {
  148. continue;
  149. }
  150. $return_img = $item;
  151. $return_img['source'] = $imgUrl;
  152. $imgUrl = htmlspecialchars($imgUrl);
  153. $imgUrl = str_replace("&amp;", "&", $imgUrl);
  154. //http开头验证
  155. if (strpos($imgUrl, "http") !== 0) {
  156. $return_img['state'] = $this->stateMap['ERROR_HTTP_LINK'];
  157. array_push($list, $return_img);
  158. continue;
  159. }
  160. //获取请求头
  161. // is_sae()
  162. if (!cmf_is_sae()) {//SAE下无效
  163. $heads = get_headers($imgUrl);
  164. //死链检测
  165. if (!(stristr($heads[0], "200") && stristr($heads[0], "OK"))) {
  166. $return_img['state'] = $this->stateMap['ERROR_DEAD_LINK'];
  167. array_push($list, $return_img);
  168. continue;
  169. }
  170. }
  171. //格式验证(扩展名验证和Content-Type验证)
  172. $fileType = strtolower(strrchr($imgUrl, '.'));
  173. if (!in_array($fileType, $config['allowFiles']) || stristr($heads['Content-Type'], "image")) {
  174. $return_img['state'] = $this->stateMap['ERROR_HTTP_CONTENTTYPE'];
  175. array_push($list, $return_img);
  176. continue;
  177. }
  178. //打开输出缓冲区并获取远程图片
  179. ob_start();
  180. $context = stream_context_create(
  181. [
  182. 'http' => [
  183. 'follow_location' => false // don't follow redirects
  184. ]
  185. ]
  186. );
  187. //请确保php.ini中的fopen wrappers已经激活
  188. readfile($imgUrl, false, $context);
  189. $img = ob_get_contents();
  190. ob_end_clean();
  191. //大小验证
  192. $uriSize = strlen($img); //得到图片大小
  193. $allowSize = 1024 * $config['maxSize'];
  194. if ($uriSize > $allowSize) {
  195. $return_img['state'] = $this->stateMap['ERROR_SIZE_EXCEED'];
  196. array_push($list, $return_img);
  197. continue;
  198. }
  199. $file = uniqid() . strrchr($imgUrl, '.');
  200. $savePath = $config['savePath'];
  201. $tmpName = $savePath . $file;
  202. //创建保存位置
  203. if (!file_exists($savePath)) {
  204. mkdir("$savePath", 0777, true);
  205. }
  206. $file_write_result = cmf_file_write($tmpName, $img);
  207. if ($file_write_result) {
  208. if (config('FILE_UPLOAD_TYPE') == 'Qiniu') {
  209. //todo qiniu code
  210. }
  211. if (config('FILE_UPLOAD_TYPE') == 'Local') {
  212. $file = $strSavePath . $file;
  213. $return_img['state'] = 'SUCCESS';
  214. $return_img['url'] = $file;
  215. array_push($list, $return_img);
  216. }
  217. } else {
  218. $return_img['state'] = $this->stateMap['ERROR_WRITE_CONTENT'];
  219. array_push($list, $return_img);
  220. }
  221. }
  222. return json_encode([
  223. 'state' => count($list) ? 'SUCCESS' : 'ERROR',
  224. 'list' => $list
  225. ]);
  226. }
  227. /**
  228. * 文件上传
  229. * @param string $fileType 文件类型
  230. * @return string
  231. */
  232. private function ueditorUpload($fileType = 'image')
  233. {
  234. $uploader = new Upload();
  235. $uploader->setStorageType("Local");
  236. $uploader->setFileType($fileType);
  237. $uploader->setFormName('upfile');
  238. $result = $uploader->upload();
  239. if ($result === false) {
  240. return json_encode([
  241. 'state' => $uploader->getError()
  242. ]);
  243. } else {
  244. return json_encode([
  245. 'state' => 'SUCCESS',
  246. 'url' => $result['url'],
  247. 'title' => $result['name'],
  248. 'original' => $result['name']
  249. ]);
  250. }
  251. }
  252. /**
  253. * 获取百度编辑器配置
  254. */
  255. private function ueditorConfig()
  256. {
  257. $config_text = preg_replace("/\/\*[\s\S]+?\*\//", "", file_get_contents(WEB_ROOT . "static/js/ueditor/config.json"));
  258. $config = json_decode($config_text, true);
  259. $upload_setting = cmf_get_upload_setting();
  260. $config['imageMaxSize'] = $upload_setting['file_types']['image']['upload_max_filesize'] * 1024;
  261. $config['imageAllowFiles'] = array_map([$this, 'ueditorExtension'], explode(",", $upload_setting['file_types']['image']['extensions']));
  262. $config['scrawlMaxSize'] = $upload_setting['file_types']['image']['upload_max_filesize'] * 1024;
  263. //
  264. $config['catcherMaxSize'] = $upload_setting['file_types']['image']['upload_max_filesize'] * 1024;
  265. $config['catcherAllowFiles'] = array_map([$this, 'ueditorExtension'], explode(",", $upload_setting['file_types']['image']['extensions']));
  266. $config['videoMaxSize'] = $upload_setting['file_types']['video']['upload_max_filesize'] * 1024;
  267. $config['videoAllowFiles'] = array_map([$this, 'ueditorExtension'], explode(",", $upload_setting['file_types']['video']['extensions']));
  268. $config['fileMaxSize'] = $upload_setting['file_types']['file']['upload_max_filesize'] * 1024;
  269. $config['fileAllowFiles'] = array_map([$this, 'ueditorExtension'], explode(",", $upload_setting['file_types']['file']['extensions']));
  270. return json_encode($config);
  271. }
  272. /**
  273. * 格式化后缀
  274. * @param $str
  275. * @return string
  276. */
  277. private function ueditorExtension($str)
  278. {
  279. return "." . trim($str, '.');
  280. }
  281. }