Keine Beschreibung
Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

AskLogic.php 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海南赞赞网络科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 小虎哥 <1105415366@qq.com>
  11. * Date: 2018-4-3
  12. */
  13. namespace app\admin\logic;
  14. use think\Model;
  15. use think\Db;
  16. /**
  17. * 逻辑定义
  18. * Class CatsLogic
  19. * @package admin\Logic
  20. */
  21. class AskLogic extends Model
  22. {
  23. private $request = null;
  24. private $data_path;
  25. private $service_url;
  26. private $upgrade_url;
  27. private $service_ey;
  28. private $planPath_pc;
  29. private $planPath_m;
  30. /**
  31. * 析构函数
  32. */
  33. function __construct() {
  34. $this->request = request();
  35. $this->service_ey = config('service_ey');
  36. $this->data_path = DATA_PATH; //
  37. // api_Service_checkVersion
  38. $tmp_str = 'L2luZGV4LnBocD9tPWFwaSZjPVVwZ3JhZGUmYT1jaGVja1RoZW1lVmVyc2lvbg==';
  39. $this->service_url = base64_decode($this->service_ey).base64_decode($tmp_str);
  40. $web_basehost = request()->host(true);
  41. if (false !== filter_var($web_basehost, FILTER_VALIDATE_IP)) {
  42. $web_basehost = tpCache('web.web_basehost');
  43. }
  44. $web_basehost = preg_replace('/^(http(s)?:)?(\/\/)?([^\/\:]*)(.*)$/i', '${4}', $web_basehost);
  45. $this->upgrade_url = $this->service_url . '&domain='.$web_basehost.'&type=theme_ask&cms_version='.getVersion().'&ip='.serverIP();
  46. $this->planPath_pc = 'template/'.TPL_THEME.'pc/';
  47. $this->planPath_m = 'template/'.TPL_THEME.'mobile/';
  48. }
  49. /**
  50. * 检测并第一次从官方同步问答中心的前台模板
  51. */
  52. public function syn_theme_ask()
  53. {
  54. error_reporting(0);//关闭所有错误报告
  55. if (!file_exists("{$this->planPath_pc}ask")) {
  56. return $this->OneKeyUpgrade();
  57. } else {
  58. return true;
  59. }
  60. }
  61. /**
  62. * 检测目录权限
  63. */
  64. public function checkAuthority($filelist = '')
  65. {
  66. /*------------------检测目录读写权限----------------------*/
  67. $filelist = htmlspecialchars_decode($filelist);
  68. $filelist = explode('<br>', $filelist);
  69. $dirs = array();
  70. $i = -1;
  71. foreach($filelist as $filename)
  72. {
  73. if (stristr($filename, $this->planPath_pc) && !file_exists($this->planPath_pc)) {
  74. continue;
  75. } else if (stristr($filename, $this->planPath_m) && !file_exists($this->planPath_m)) {
  76. continue;
  77. }
  78. $tfilename = $filename;
  79. $curdir = $this->GetDirName($tfilename);
  80. if (empty($curdir)) {
  81. continue;
  82. }
  83. if( !isset($dirs[$curdir]) )
  84. {
  85. $dirs[$curdir] = $this->TestIsFileDir($curdir);
  86. }
  87. if($dirs[$curdir]['isdir'] == FALSE)
  88. {
  89. continue;
  90. }
  91. else {
  92. $dirs[$curdir] = $this->TestIsFileDir($curdir);
  93. }
  94. $i++;
  95. }
  96. $is_pass = true;
  97. $msg = '检测通过';
  98. if($i > -1)
  99. {
  100. $n = 0;
  101. $dirinfos = '';
  102. foreach($dirs as $curdir)
  103. {
  104. $dirinfos .= $curdir['name']."&nbsp;&nbsp;状态:";
  105. if ($curdir['writeable']) {
  106. $dirinfos .= "[√正常]";
  107. } else {
  108. $is_pass = false;
  109. $n++;
  110. $dirinfos .= "<font color='red'>[×不可写]</font>";
  111. }
  112. $dirinfos .= "<br />";
  113. }
  114. $title = "本次升级需要在下面文件夹写入更新文件,已检测站点有 <font color='red'>{$n}</font> 处没有写入权限:<br />";
  115. $title .= "<font color='red'>问题分析(如有问题,请咨询技术支持):<br />";
  116. $title .= "1、检查站点目录的用户组与所有者,禁止是 root ;<br />";
  117. $title .= "2、检查站点目录的读写权限,一般权限值是 0755 ;<br />";
  118. $title .= "</font>涉及更新目录列表如下:<br />";
  119. $msg = $title . $dirinfos;
  120. }
  121. /*------------------end----------------------*/
  122. if (true === $is_pass) {
  123. return ['code'=>1, 'msg'=>$msg];
  124. } else {
  125. return ['code'=>0, 'msg'=>$msg, 'data'=>['code'=>1]];
  126. }
  127. }
  128. /**
  129. * 检查是否有更新包
  130. * @return type 提示语
  131. */
  132. public function checkVersion() {
  133. //error_reporting(0);//关闭所有错误报告
  134. $allow_url_fopen = ini_get('allow_url_fopen');
  135. if (!$allow_url_fopen) {
  136. return ['code' => 1, 'msg' => "<font color='red'>请联系空间商(设置 php.ini 中参数 allow_url_fopen = 1)</font>"];
  137. }
  138. $url = $this->upgrade_url;
  139. $serviceVersionList = @httpRequest($url);
  140. if (false === $serviceVersionList) {
  141. $context = stream_context_set_default(array('http' => array('timeout' => 3,'method'=>'GET')));
  142. $serviceVersionList = @file_get_contents($url,false,$context);
  143. }
  144. $serviceVersionList = json_decode($serviceVersionList,true);
  145. if(!empty($serviceVersionList))
  146. {
  147. $upgradeArr = array();
  148. $introStr = '';
  149. $upgradeStr = '';
  150. foreach ($serviceVersionList as $key => $val) {
  151. $upgrade = !empty($val['upgrade']) ? $val['upgrade'] : array();
  152. $upgradeArr = array_merge($upgradeArr, $upgrade);
  153. $introStr .= '<br>'.filter_line_return($val['intro'], '<br>');
  154. }
  155. $upgradeArr = array_unique($upgradeArr);
  156. foreach ($upgradeArr as $key => $val) {
  157. if (stristr($val, $this->planPath_pc) && !file_exists($this->planPath_pc)) {
  158. unset($upgradeArr[$key]);
  159. } else if (stristr($val, $this->planPath_m) && !file_exists($this->planPath_m)) {
  160. unset($upgradeArr[$key]);
  161. }
  162. }
  163. $upgradeStr = implode('<br>', $upgradeArr); // 升级提示需要覆盖哪些文件
  164. $introArr = explode('<br>', $introStr);
  165. $introStr = '更新日志:';
  166. foreach ($introArr as $key => $val) {
  167. if (empty($val)) {
  168. continue;
  169. }
  170. $introStr .= "<br>{$key}、".$val;
  171. }
  172. $lastupgrade = $serviceVersionList[count($serviceVersionList) - 1];
  173. if (!empty($lastupgrade['upgrade_title'])) {
  174. $introStr .= '<br>'.$lastupgrade['upgrade_title'];
  175. }
  176. $lastupgrade['intro'] = htmlspecialchars_decode($introStr);
  177. $lastupgrade['upgrade'] = htmlspecialchars_decode($upgradeStr); // 升级提示需要覆盖哪些文件
  178. /*升级公告*/
  179. if (!empty($lastupgrade['notice'])) {
  180. $lastupgrade['notice'] = htmlspecialchars_decode($lastupgrade['notice']) . '<br>';
  181. }
  182. /*--end*/
  183. return ['code' => 2, 'msg' => $lastupgrade];
  184. }
  185. return ['code' => 1, 'msg' => '已是最新版'];
  186. }
  187. /**
  188. * 检查是否有更新包
  189. * @return type 提示语
  190. */
  191. public function OneKeyUpgrade() {
  192. $allow_url_fopen = ini_get('allow_url_fopen');
  193. if (!$allow_url_fopen) {
  194. return ['code' => 0, 'msg' => "请联系空间商,设置 php.ini 中参数 allow_url_fopen = 1"];
  195. }
  196. if (!extension_loaded('zip')) {
  197. return ['code' => 0, 'msg' => "请联系空间商,开启 php.ini 中的php-zip扩展"];
  198. }
  199. $serviceVersionList = @httpRequest($this->upgrade_url);
  200. if (false === $serviceVersionList) {
  201. $serviceVersionList = @file_get_contents($this->upgrade_url);
  202. }
  203. $serviceVersionList = json_decode($serviceVersionList,true);
  204. if (empty($serviceVersionList)) {
  205. return ['code' => 0, 'msg' => "没找到模板包信息"];
  206. } else if (isset($serviceVersionList['code']) && empty($serviceVersionList['code'])) {
  207. $icon = !empty($serviceVersionList['icon']) ? $serviceVersionList['icon'] : 2;
  208. return ['code' => 0, 'msg' => $serviceVersionList['msg'], 'icon'=>$icon];
  209. }
  210. /*最新更新版本信息*/
  211. $lastServiceVersion = $serviceVersionList[count($serviceVersionList) - 1];
  212. /*--end*/
  213. /*批量下载更新包*/
  214. $upgradeArr = array(); // 更新的文件列表
  215. $folderName = 'ask-'.$lastServiceVersion['key_num'];
  216. foreach ($serviceVersionList as $key => $val) {
  217. // 下载更新包
  218. $result = $this->downloadFile($val['down_url'], $val['file_md5']);
  219. if (!isset($result['code']) || $result['code'] != 1) {
  220. return $result;
  221. }
  222. /*第一个循环执行的业务*/
  223. if ($key == 0) {
  224. /*解压到最后一个更新包的文件夹*/
  225. $lastDownFileName = explode('/', $lastServiceVersion['down_url']);
  226. $lastDownFileName = end($lastDownFileName);
  227. $folderName = 'ask-'.str_replace(".zip", "", $lastDownFileName); // 文件夹
  228. /*--end*/
  229. /*解压之前,删除已重复的文件夹*/
  230. delFile($this->data_path.'backup'.DS.'theme'.DS.$folderName);
  231. /*--end*/
  232. }
  233. /*--end*/
  234. $downFileName = explode('/', $val['down_url']);
  235. $downFileName = 'ask-'.end($downFileName);
  236. /*解压文件*/
  237. $zip = new \ZipArchive();//新建一个ZipArchive的对象
  238. if ($zip->open($this->data_path.'backup'.DS.'theme'.DS.$downFileName) != true) {
  239. return ['code' => 0, 'msg' => "模板包读取失败!"];
  240. }
  241. $zip->extractTo($this->data_path.'backup'.DS.'theme'.DS.$folderName.DS);//假设解压缩到在当前路径下backup文件夹内
  242. $zip->close();//关闭处理的zip文件
  243. /*--end*/
  244. /*更新的文件列表*/
  245. $upgrade = !empty($val['upgrade']) ? $val['upgrade'] : array();
  246. $upgradeArr = array_merge($upgradeArr, $upgrade);
  247. /*--end*/
  248. }
  249. /*--end*/
  250. /*将多个更新包重新组建一个新的完全更新包*/
  251. $upgradeArr = array_unique($upgradeArr); // 移除文件列表里重复的文件
  252. $serviceVersion = $lastServiceVersion;
  253. $serviceVersion['upgrade'] = $upgradeArr;
  254. /*--end*/
  255. /*升级之前,备份涉及的源文件*/
  256. $upgrade = $serviceVersion['upgrade'];
  257. if (!empty($upgrade) && is_array($upgrade)) {
  258. foreach ($upgrade as $key => $val) {
  259. $source_file = ROOT_PATH.$val;
  260. if (file_exists($source_file)) {
  261. $destination_file = $this->data_path.'backup'.DS.'theme'.DS.$folderName.'_www'.DS.$val;
  262. tp_mkdir(dirname($destination_file));
  263. $copy_bool = @copy($source_file, $destination_file);
  264. if (false == $copy_bool) {
  265. return ['code' => 0, 'msg' => "备份文件失败,请检查所有目录是否有读写权限"];
  266. }
  267. }
  268. }
  269. }
  270. /*--end*/
  271. // 递归复制文件夹
  272. $copy_data = $this->recurse_copy($this->data_path.'backup'.DS.'theme'.DS.$folderName, rtrim(ROOT_PATH, DS), $folderName);
  273. /*删除下载的模板包*/
  274. $ziplist = glob($this->data_path.'backup'.DS.'theme'.DS.'ask-*.zip');
  275. @array_map('unlink', $ziplist);
  276. /*--end*/
  277. // 推送回服务器 记录升级成功
  278. $this->UpgradeLog($serviceVersion['key_num']);
  279. return ['code' => $copy_data['code'], 'msg' => "更新模板成功{$copy_data['msg']}"];
  280. }
  281. /**
  282. * 自定义函数递归的复制带有多级子目录的目录
  283. * 递归复制文件夹
  284. *
  285. * @param string $src 原目录
  286. * @param string $dst 复制到的目录
  287. * @param string $folderName 存放升级包目录名称
  288. * @return string
  289. */
  290. //参数说明:
  291. //自定义函数递归的复制带有多级子目录的目录
  292. private function recurse_copy($src, $dst, $folderName)
  293. {
  294. static $badcp = 0; // 累计覆盖失败的文件总数
  295. static $n = 0; // 累计执行覆盖的文件总数
  296. static $total = 0; // 累计更新的文件总数
  297. $dir = opendir($src);
  298. /*pc和mobile目录存在的情况下,才拷贝会员模板到相应的pc或mobile里*/
  299. $dst_tmp = str_replace('\\', '/', $dst);
  300. $dst_tmp = rtrim($dst_tmp, '/').'/';
  301. if (stristr($dst_tmp, $this->planPath_pc) && file_exists($this->planPath_pc)) {
  302. tp_mkdir($dst);
  303. } else if (stristr($dst_tmp, $this->planPath_m) && file_exists($this->planPath_m)) {
  304. tp_mkdir($dst);
  305. }
  306. /*--end*/
  307. while (false !== $file = readdir($dir)) {
  308. if (($file != '.') && ($file != '..')) {
  309. if (is_dir($src . '/' . $file)) {
  310. $needle = '/template/'.TPL_THEME;
  311. $needle = rtrim($needle, '/');
  312. $dstfile = $dst . '/' . $file;
  313. if (!stristr($dstfile, $needle)) {
  314. $dstfile = str_replace('/template', $needle, $dstfile);
  315. }
  316. $this->recurse_copy($src . '/' . $file, $dstfile, $folderName);
  317. }
  318. else {
  319. if (file_exists($src . DIRECTORY_SEPARATOR . $file)) {
  320. /*pc和mobile目录存在的情况下,才拷贝会员模板到相应的pc或mobile里*/
  321. $rs = true;
  322. $src_tmp = str_replace('\\', '/', $src . DIRECTORY_SEPARATOR . $file);
  323. if (stristr($src_tmp, $this->planPath_pc) && !file_exists($this->planPath_pc)) {
  324. continue;
  325. } else if (stristr($src_tmp, $this->planPath_m) && !file_exists($this->planPath_m)) {
  326. continue;
  327. }
  328. /*--end*/
  329. $rs = @copy($src . DIRECTORY_SEPARATOR . $file, $dst . DIRECTORY_SEPARATOR . $file);
  330. if($rs) {
  331. $n++;
  332. @unlink($src . DIRECTORY_SEPARATOR . $file);
  333. } else {
  334. $n++;
  335. $badcp++;
  336. }
  337. } else {
  338. $n++;
  339. }
  340. $total++;
  341. }
  342. }
  343. }
  344. closedir($dir);
  345. $code = 1;
  346. $msg = '!';
  347. if($badcp > 0)
  348. {
  349. $code = 2;
  350. $msg = ",其中失败 <font color='red'>{$badcp}</font> 个文件,<br />请从模板包目录[<font color='red'>data/backup/theme/{$folderName}</font>]中的取出全部文件覆盖到根目录,完成手工覆盖。";
  351. }
  352. $this->copy_speed($n, $total);
  353. return ['code'=>$code, 'msg'=>$msg];
  354. }
  355. /**
  356. * 复制文件进度
  357. */
  358. private function copy_speed($n, $total)
  359. {
  360. $data = false;
  361. if ($n < $total) {
  362. $this->copy_speed($n, $total);
  363. } else {
  364. $data = true;
  365. }
  366. return $data;
  367. }
  368. /**
  369. * @param type $fileUrl 下载文件地址
  370. * @param type $md5File 文件MD5 加密值 用于对比下载是否完整
  371. * @return string 错误或成功提示
  372. */
  373. private function downloadFile($fileUrl,$md5File)
  374. {
  375. $downFileName = explode('/', $fileUrl);
  376. $downFileName = 'ask-'.end($downFileName);
  377. $saveDir = $this->data_path.'backup'.DS.'theme'.DS.$downFileName; // 保存目录
  378. tp_mkdir(dirname($saveDir));
  379. $content = @httpRequest($fileUrl);
  380. if (false === $content) {
  381. $content = @file_get_contents($fileUrl, 0, null, 0, 1);
  382. }
  383. if(!$content){
  384. return ['code' => 0, 'msg' => '官方问答模板包不存在']; // 文件存在直接退出
  385. }
  386. if (!stristr($fileUrl, 'https://service')) {
  387. $ch = curl_init($fileUrl);
  388. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
  389. curl_setopt($ch, CURLOPT_BINARYTRANSFER,1);
  390. $file = curl_exec ($ch);
  391. curl_close ($ch);
  392. } else {
  393. $file = httpRequest($fileUrl);
  394. }
  395. if (preg_match('#__HALT_COMPILER()#i', $file)) {
  396. return ['code' => 0, 'msg' => '问答模板包损坏,请联系官方客服!'];
  397. }
  398. $fp = fopen($saveDir,'w');
  399. fwrite($fp, $file);
  400. fclose($fp);
  401. if(!eyPreventShell($saveDir) || !file_exists($saveDir) || $md5File != md5_file($saveDir))
  402. {
  403. return ['code' => 0, 'msg' => '下载保存问答模板包失败,请检查所有目录的权限以及用户组不能为root'];
  404. }
  405. return ['code' => 1, 'msg' => '下载成功'];
  406. }
  407. // 升级记录 log 日志
  408. private function UpgradeLog($to_key_num){
  409. $serial_number = DEFAULT_SERIALNUMBER;
  410. $constsant_path = APP_PATH.MODULE_NAME.'/conf/constant.php';
  411. if (file_exists($constsant_path)) {
  412. require_once($constsant_path);
  413. defined('SERIALNUMBER') && $serial_number = SERIALNUMBER;
  414. }
  415. $mysqlinfo = \think\Db::query("SELECT VERSION() as version");
  416. $mysql_version = $mysqlinfo[0]['version'];
  417. $values = array(
  418. 'type' => 'theme_ask',
  419. 'domain'=>request()->host(), //用户域名
  420. 'key_num'=>'v1.0.0', // 用户版本号
  421. 'to_key_num'=>$to_key_num, // 用户要升级的版本号
  422. 'add_time'=>time(), // 升级时间
  423. 'serial_number'=>$serial_number,
  424. 'ip' => GetHostByName($_SERVER['SERVER_NAME']),
  425. 'phpv' => phpversion(),
  426. 'mysql_version' => $mysql_version,
  427. 'web_server' => $_SERVER['SERVER_SOFTWARE'],
  428. );
  429. // api_Service_upgradeLog
  430. $tmp_str = 'L2luZGV4LnBocD9tPWFwaSZjPVVwZ3JhZGUmYT11cGdyYWRlTG9nJg==';
  431. $url = base64_decode($this->service_ey).base64_decode($tmp_str).http_build_query($values);
  432. @httpRequest($url);
  433. }
  434. /**
  435. * 获取文件的目录路径
  436. * @param string $filename 文件路径+文件名
  437. * @return string
  438. */
  439. private function GetDirName($filename)
  440. {
  441. $dirname = preg_replace("#[\\\\\/]{1,}#", '/', $filename);
  442. $dirname = preg_replace("#([^\/]*)$#", '', $dirname);
  443. return $dirname;
  444. }
  445. /**
  446. * 测试目录路径是否有读写权限
  447. * @param string $dirname 文件目录路径
  448. * @return array
  449. */
  450. private function TestIsFileDir($dirname)
  451. {
  452. $dirs = array('name'=>'', 'isdir'=>FALSE, 'writeable'=>FALSE);
  453. $dirs['name'] = $dirname;
  454. tp_mkdir($dirname);
  455. if(is_dir($dirname))
  456. {
  457. $dirs['isdir'] = TRUE;
  458. $dirs['writeable'] = $this->TestWriteAble($dirname);
  459. }
  460. return $dirs;
  461. }
  462. /**
  463. * 测试目录路径是否有写入权限
  464. * @param string $d 目录路劲
  465. * @return boolean
  466. */
  467. private function TestWriteAble($d)
  468. {
  469. $tfile = '_eyout.txt';
  470. $fp = @fopen($d.$tfile,'w');
  471. if(!$fp) {
  472. return false;
  473. }
  474. else {
  475. fclose($fp);
  476. $rs = @unlink($d.$tfile);
  477. return true;
  478. }
  479. }
  480. }