心理咨询网
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.

UpgradeController.php 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408
  1. <?php
  2. /**
  3. * @copyright (C)2016-2099 Hnaoyun Inc.
  4. * @author XingMeng
  5. * @email hnxsh@foxmail.com
  6. * @date 2018年8月14日
  7. * 在线更新
  8. */
  9. namespace app\admin\controller\system;
  10. use core\basic\Controller;
  11. use core\basic\Model;
  12. class UpgradeController extends Controller
  13. {
  14. // 服务器地址
  15. private $server = 'https://www.pbootcms.com';
  16. // 更新分支
  17. private $branch;
  18. // 强制同步文件
  19. private $force;
  20. // 修改版本
  21. private $revise;
  22. // 文件列表
  23. public $files = array();
  24. public function __construct()
  25. {
  26. error_reporting(0);
  27. $this->branch = $this->config('upgrade_branch') == '3.X.dev' ? '3.X.dev' : '3.X';
  28. $this->force = $this->config('upgrade_force') ?: 0;
  29. $this->revise = $this->config('revise_version') ?: 0;
  30. }
  31. public function index()
  32. {
  33. switch (get('action')) {
  34. case 'local':
  35. $upfile = $this->local();
  36. break;
  37. default:
  38. $upfile = array();
  39. }
  40. $this->assign('upfile', $upfile);
  41. $this->assign('branch', $this->branch);
  42. $this->assign('force', $this->force);
  43. $this->assign('revise', $this->revise);
  44. $this->assign('snuser', $this->config('sn_user') ?: 0);
  45. $this->assign('site', get_http_url());
  46. $this->display('system/upgrade.html');
  47. }
  48. public function checkCache(){
  49. $now = time();
  50. $cache = $_SESSION['check_cache'];
  51. if(!$cache){
  52. $_SESSION['check_cache'] = time();
  53. json(1,'');
  54. }else{
  55. $sub = $now - $cache;
  56. if($sub > 3600){
  57. $_SESSION['check_cache'] = time();
  58. json(1,'');
  59. }else{
  60. json(0,'');
  61. }
  62. }
  63. }
  64. // 检查更新
  65. public function check()
  66. {
  67. // 清理目录,检查下载目录及备份目录
  68. path_delete(RUN_PATH . '/upgrade', true);
  69. if (! check_dir(RUN_PATH . '/upgrade', true)) {
  70. json(0, '目录写入权限不足,无法正常升级!' . RUN_PATH . '/upgrade');
  71. }
  72. check_dir(DOC_PATH . STATIC_DIR . '/backup/upgrade', true);
  73. $files = $this->getServerList();
  74. $db = get_db_type();
  75. foreach ($files as $key => $value) {
  76. // 过滤掉相对路径
  77. $value->path = preg_replace_r('{\.\.(\/|\\\\)}', '', $value->path);
  78. $file = ROOT_PATH . $value->path;
  79. if (@md5_file($file) != $value->md5) {
  80. // 筛选数据库更新脚本
  81. if (preg_match('/([\w]+)-([\w\.]+)-update\.sql/i', $file, $matches)) {
  82. if ($matches[1] != $db || ! $this->compareVersion($matches[2], APP_VERSION . '.' . RELEASE_TIME . '.' . $this->revise)) {
  83. continue;
  84. }
  85. }
  86. if (file_exists($file)) {
  87. $files[$key]->type = '<span style="color:Red">覆盖</span>';
  88. $files[$key]->ltime = date('Y-m-d H:i:s', filemtime($file));
  89. } else {
  90. $files[$key]->type = '新增';
  91. $files[$key]->ltime = '无';
  92. }
  93. $files[$key]->ctime = date('Y-m-d H:i:s', $files[$key]->ctime);
  94. $upfile[] = $files[$key];
  95. }
  96. }
  97. if (! $upfile) {
  98. json(1, '您的系统无任何文件需要更新!');
  99. } else {
  100. json(1, $upfile);
  101. }
  102. }
  103. // 执行下载
  104. public function down()
  105. {
  106. if (! ! $list = get('list')) {
  107. if (! is_array($list)) { // 单个文件转换为数组
  108. $list = array(
  109. $list
  110. );
  111. }
  112. $len = count($list) ?: 0;
  113. foreach ($list as $value) {
  114. // 过滤掉相对路径
  115. $value = preg_replace_r('{\.\.(\/|\\\\)}', '', $value);
  116. // 本地存储路径
  117. $path = RUN_PATH . '/upgrade' . $value;
  118. // 自动创建目录
  119. if (! check_dir(dirname($path), true)) {
  120. json(0, '目录写入权限不足,无法下载升级文件!' . dirname($path));
  121. }
  122. // 定义执行下载的类型
  123. $types = '.zip|.rar|.doc|.docx|.ppt|.pptx|.xls|.xlsx|.chm|.ttf|.otf|';
  124. $pathinfo = explode(".", basename($path));
  125. $ext = end($pathinfo); // 获取扩展
  126. if (preg_match('/\.' . $ext . '\|/i', $types)) {
  127. $result = $this->getServerDown('/release/' . $this->branch . $value, $path);
  128. } else {
  129. $result = $this->getServerFile($value, $path);
  130. }
  131. }
  132. if ($len == 1) {
  133. json(1, "更新文件 " . basename($value) . " 下载成功!");
  134. } else {
  135. json(1, "更新文件" . basename($value) . "等文件全部下载成功!");
  136. }
  137. } else {
  138. json(0, '请选择要下载的文件!');
  139. }
  140. }
  141. // 执行更新
  142. public function update()
  143. {
  144. if ($_POST) {
  145. if (! ! $list = post('list')) {
  146. $list = explode(',', $list);
  147. $backdir = date('YmdHis');
  148. // 分离文件
  149. foreach ($list as $value) {
  150. // 过滤掉相对路径
  151. $value = preg_replace_r('{\.\.(\/|\\\\)}', '', $value);
  152. if (stripos($value, '/script/') === 0 && preg_match('/\.sql$/i', $value)) {
  153. $sqls[] = $value;
  154. } else {
  155. $path = RUN_PATH . '/upgrade' . $value;
  156. $des_path = ROOT_PATH . $value;
  157. $back_path = DOC_PATH . STATIC_DIR . '/backup/upgrade/' . $backdir . $value;
  158. if (! check_dir(dirname($des_path), true)) {
  159. json(0, '目录写入权限不足,无法正常升级!' . dirname($des_path));
  160. }
  161. if (file_exists($des_path)) { // 文件存在时执行备份
  162. check_dir(dirname($back_path), true);
  163. copy($des_path, $back_path);
  164. }
  165. // 如果后台入口文件修改过名字,则自动适配
  166. if (stripos($path, 'admin.php') !== false && stripos($_SERVER['SCRIPT_FILENAME'], 'admin.php') === false) {
  167. if (file_exists($_SERVER['SCRIPT_FILENAME'])) {
  168. $des_path = $_SERVER['SCRIPT_FILENAME'];
  169. }
  170. }
  171. $files[] = array(
  172. 'sfile' => $path,
  173. 'dfile' => $des_path
  174. );
  175. }
  176. }
  177. // 更新数据库
  178. if (isset($sqls)) {
  179. $db = new DatabaseController();
  180. switch (get_db_type()) {
  181. case 'sqlite':
  182. copy(DOC_PATH . $this->config('database.dbname'), DOC_PATH . STATIC_DIR . '/backup/sql/' . date('YmdHis') . '_' . basename($this->config('database.dbname')));
  183. break;
  184. case 'mysql':
  185. $db->backupDB();
  186. break;
  187. }
  188. sort($sqls); // 排序
  189. foreach ($sqls as $value) {
  190. $path = RUN_PATH . '/upgrade' . $value;
  191. if (file_exists($path)) {
  192. $sql = file_get_contents($path);
  193. if (! $this->upsql($sql)) {
  194. $this->log("数据库 $value 更新失败!");
  195. json(0, "数据库" . basename($value) . " 更新失败!");
  196. }
  197. } else {
  198. json(0, "数据库文件" . basename($value) . "不存在!");
  199. }
  200. }
  201. }
  202. // 替换文件
  203. if (isset($files)) {
  204. foreach ($files as $value) {
  205. if (! copy($value['sfile'], $value['dfile'])) {
  206. $this->log("文件 " . $value['dfile'] . " 更新失败!");
  207. json(0, "文件 " . basename($value['dfile']) . " 更新失败,请重试!");
  208. }
  209. }
  210. }
  211. // 清理缓存
  212. path_delete(RUN_PATH . '/upgrade', true);
  213. path_delete(RUN_PATH . '/cache');
  214. path_delete(RUN_PATH . '/complite');
  215. path_delete(RUN_PATH . '/config');
  216. $this->log("系统更新成功!");
  217. json(1, '系统更新成功!');
  218. } else {
  219. json(0, '请选择要更新的文件!');
  220. }
  221. }
  222. }
  223. // 缓存文件
  224. private function local()
  225. {
  226. $files = $this->getLoaclList(RUN_PATH . '/upgrade');
  227. $files = json_decode(json_encode($files));
  228. foreach ($files as $key => $value) {
  229. $file = ROOT_PATH . $value->path;
  230. if (file_exists($file)) {
  231. $files[$key]->type = '<span style="color:Red">覆盖</span>';
  232. $files[$key]->ltime = date('Y-m-d H:i:s', filemtime($file));
  233. } else {
  234. $files[$key]->type = '新增';
  235. $files[$key]->ltime = '无';
  236. }
  237. $files[$key]->ctime = date('Y-m-d H:i:s', $files[$key]->ctime);
  238. $upfile[] = $files[$key];
  239. }
  240. return $upfile;
  241. }
  242. // 执行更新数据库
  243. private function upsql($sql)
  244. {
  245. $sql = explode(';', $sql);
  246. $model = new Model();
  247. foreach ($sql as $value) {
  248. $value = trim($value);
  249. if ($value) {
  250. $model->amd($value);
  251. }
  252. }
  253. return true;
  254. }
  255. // 获取列表
  256. private function getServerList()
  257. {
  258. $param = array(
  259. 'version' => APP_VERSION . '.' . RELEASE_TIME . '.' . $this->revise,
  260. 'branch' => $this->branch,
  261. 'force' => $this->force,
  262. 'site' => get_http_url(),
  263. 'snuser' => $this->config('sn_user')
  264. );
  265. $url = $this->server . '/index.php?p=/upgrade/getlist&' . http_build_query($param);
  266. if (! ! $rs = json_decode(get_url($url, '', '', true))) {
  267. if ($rs->code) {
  268. if (is_array($rs->data)) {
  269. return $rs->data;
  270. } else {
  271. json(1, $rs->data);
  272. }
  273. } else {
  274. json(0, $rs->data);
  275. }
  276. } else {
  277. $this->log('连接更新服务器发生错误,请稍后再试!');
  278. json(0, '连接更新服务器发生错误,请稍后再试!');
  279. }
  280. }
  281. // 获取文件
  282. private function getServerFile($source, $des)
  283. {
  284. $url = $this->server . '/index.php?p=/upgrade/getFile&branch=' . $this->branch;
  285. $data['path'] = $source;
  286. $file = basename($source);
  287. if (! ! $rs = json_decode(get_url($url, $data, '', true))) {
  288. if ($rs->code) {
  289. if (! file_put_contents($des, base64_decode($rs->data))) {
  290. $this->log("更新文件 " . $file . " 下载失败!");
  291. json(0, "更新文件 " . $file . " 下载失败!");
  292. } else {
  293. return true;
  294. }
  295. } else {
  296. json(0, $rs->data);
  297. }
  298. } else {
  299. $this->log("更新文件 " . $file . " 获取失败!");
  300. json(0, "更新文件 " . $file . " 获取失败!");
  301. }
  302. }
  303. // 获取非文本文件
  304. private function getServerDown($source, $des)
  305. {
  306. $url = $this->server . $source;
  307. $file = basename($source);
  308. if (($sfile = fopen($url, "rb")) && ($dfile = fopen($des, "wb"))) {
  309. while (! feof($sfile)) {
  310. $fwrite = fwrite($dfile, fread($sfile, 1024 * 8), 1024 * 8);
  311. if ($fwrite === false) {
  312. $this->log("更新文件 " . $file . " 下载失败!");
  313. json(0, "更新文件 " . $file . " 下载失败!");
  314. }
  315. }
  316. if ($sfile) {
  317. fclose($sfile);
  318. }
  319. if ($dfile) {
  320. fclose($dfile);
  321. }
  322. return true;
  323. } else {
  324. $this->log("更新文件 " . $file . " 获取失败!");
  325. json(0, "更新文件 " . $file . " 获取失败!");
  326. }
  327. }
  328. // 获取文件列表
  329. private function getLoaclList($path)
  330. {
  331. $files = scandir($path);
  332. foreach ($files as $value) {
  333. if ($value != '.' && $value != '..') {
  334. if (is_dir($path . '/' . $value)) {
  335. $this->getLoaclList($path . '/' . $value);
  336. } else {
  337. $file = $path . '/' . $value;
  338. // 避免中文乱码
  339. if (! mb_check_encoding($file, 'utf-8')) {
  340. $out_path = mb_convert_encoding($file, 'UTF-8', 'GBK');
  341. } else {
  342. $out_path = $file;
  343. }
  344. $out_path = str_replace(RUN_PATH . '/upgrade', '', $out_path);
  345. $this->files[] = array(
  346. 'path' => $out_path,
  347. 'md5' => md5_file($file),
  348. 'ctime' => filemtime($file)
  349. );
  350. }
  351. }
  352. }
  353. return $this->files;
  354. }
  355. // 比较程序本号
  356. private function compareVersion($sv, $cv)
  357. {
  358. if (empty($sv) || $sv == $cv) {
  359. return 0;
  360. }
  361. $sv = explode('.', $sv);
  362. $cv = explode('.', $cv);
  363. $len = count($sv) > count($cv) ? count($sv) : count($cv);
  364. for ($i = 0; $i < $len; $i ++) {
  365. $n1 = $sv[$i] or 0;
  366. $n2 = $cv[$i] or 0;
  367. if ($n1 > $n2) {
  368. return 1;
  369. } elseif ($n1 < $n2) {
  370. return 0;
  371. }
  372. }
  373. return 0;
  374. }
  375. }