No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

MemberLogic.php 20KB

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