index.php 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | 萤火商城系统 [ 致力于通过产品和服务,帮助商家高效化开拓市场 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2017~2021 https://www.yiovo.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed 这不是一个自由软件,不允许对程序代码以任何形式任何目的的再发行
  8. // +----------------------------------------------------------------------
  9. // | Author: 萤火科技 <admin@yiovo.com>
  10. // +----------------------------------------------------------------------
  11. header('Content-Type:text/html; charset=utf-8');
  12. // 检测php版本号
  13. if (phpversion() < '7.1') {
  14. exit('很抱歉,由于您的PHP版本过低,不能安装本软件,为了系统功能全面可用,请升级到PHP7.1或更高版本再安装,谢谢!');
  15. }
  16. // 不限制响应时间
  17. // error_reporting(0);
  18. set_time_limit(0);
  19. // 设置系统路径
  20. define('IN_INSTALL', true);
  21. define('INSTALL_PATH', str_replace('\\', '/', dirname(__FILE__)));
  22. define('ROOT_PATH', dirname(INSTALL_PATH, 2));
  23. // 版权信息设置
  24. $cfg_copyright = '© 2018-2022 YIOVO.COM';
  25. // 获取当前步骤
  26. $s = getStep();
  27. // 提示已经安装
  28. if (is_file(INSTALL_PATH . '/install.lock') && $s != md5('done')) {
  29. require_once(INSTALL_PATH . '/templates/step_5.php');
  30. exit();
  31. }
  32. // 执行相应操作
  33. $GLOBALS['isNext'] = true;
  34. // 获取当前步骤
  35. function getStep()
  36. {
  37. $s1 = $_GET['s'] ?? 0;
  38. // 初始化参数
  39. $s2 = $_POST['s'] ?? 0;
  40. // 如果有GET值则覆盖POST值
  41. if ($s1 > 0 && in_array($s1, [1, 63832, md5('done')])) {
  42. $s2 = $s1;
  43. }
  44. return $s2;
  45. }
  46. // 协议说明
  47. if ($s == 0) {
  48. require_once(INSTALL_PATH . '/templates/step_0.php');
  49. exit();
  50. }
  51. // 环境检测
  52. if ($s == 1) {
  53. // 获取检测的路径数据
  54. $iswrite_array = getIsWriteArray();
  55. // 获取检测的函数数据
  56. $exists_array = getExistsFuncArray();
  57. // 获取扩展要求数据
  58. $extendArray = getExtendArray();
  59. // 引入环境检测html
  60. require_once(INSTALL_PATH . '/templates/step_1.php');
  61. exit();
  62. }
  63. // 配置文件
  64. if ($s == 2) {
  65. require_once(INSTALL_PATH . '/templates/step_2.php');
  66. exit();
  67. }
  68. // 正在安装
  69. if ($s == 3) {
  70. require_once(INSTALL_PATH . '/templates/step_3.php');
  71. if ($_POST['s'] == 3) {
  72. // 初始化信息
  73. $dbhost = $_POST['dbhost'] ?? '';
  74. $dbname = $_POST['dbname'] ?? '';
  75. $dbuser = $_POST['dbuser'] ?? '';
  76. $dbpwd = $_POST['dbpwd'] ?? '';
  77. $dbport = $_POST['dbport'] ?? 3306;
  78. $testdata = $_POST['testdata'] ?? '';
  79. // 连接证数据库
  80. try {
  81. $dsn = "mysql:host={$dbhost};port={$dbport};charset=utf8";
  82. $pdo = new PDO($dsn, $dbuser, $dbpwd);
  83. $pdo->query("SET NAMES utf8"); // 设置数据库编码
  84. } catch (Exception $e) {
  85. insError('数据库连接错误,请检查!');
  86. }
  87. // 查询数据库
  88. $res = $pdo->query('show Databases');
  89. // 遍历所有数据库,存入数组
  90. $dbnameArr = [];
  91. foreach ($res->fetchAll(PDO::FETCH_ASSOC) as $row) {
  92. $dbnameArr[] = $row['Database'];
  93. }
  94. // 检查数据库是否存在,没有则创建数据库
  95. if (!in_array(trim($dbname), $dbnameArr)) {
  96. if (!$pdo->exec("CREATE DATABASE `$dbname`")) {
  97. insError("创建数据库失败,请检查权限或联系管理员!");
  98. }
  99. }
  100. // 数据库创建完成,开始连接
  101. $pdo->query("USE `$dbname`");
  102. // 取出.env模板内容
  103. $config_str = readDataFile('.env.tpl');
  104. // 进行替换
  105. $config_str = str_replace('~db_host~', $dbhost, $config_str);
  106. $config_str = str_replace('~db_name~', $dbname, $config_str);
  107. $config_str = str_replace('~db_user~', $dbuser, $config_str);
  108. $config_str = str_replace('~db_pwd~', $dbpwd, $config_str);
  109. $config_str = str_replace('~db_port~', $dbport, $config_str);
  110. $config_str = str_replace('~db_charset~', 'utf8', $config_str);
  111. // 将替换后的内容写入.env文件
  112. $fp = fopen(ROOT_PATH . '/.env', 'w');
  113. fwrite($fp, $config_str);
  114. fclose($fp);
  115. // 防止浏览器缓存
  116. $buffer = ini_get('output_buffering');
  117. echo str_repeat(' ', $buffer + 1);
  118. insInfo("数据库连接文件创建完成!");
  119. ob_flush();
  120. flush();
  121. // 创建表结构
  122. $tbstruct = readDataFile('install_struct.sql');
  123. $pdo->exec(trim($tbstruct));
  124. insInfo("数据库结构导入完成!");
  125. ob_flush();
  126. flush();
  127. // 导入其他安装数据
  128. $data_str = readDataFile('install_data.sql');
  129. $pdo->exec(trim($data_str));
  130. insInfo("商城默认数据导入完成!");
  131. ob_flush();
  132. flush();
  133. // 查看是否需要安装测试数据
  134. if ($testdata == 'true') {
  135. insInfo("正在加载测试数据!");
  136. ob_flush();
  137. flush();
  138. $sqlstr_file = readDataFile('install_testdata.sql');
  139. $pdo->exec(trim($sqlstr_file));
  140. insInfo("测试数据导入完成!");
  141. ob_flush();
  142. flush();
  143. }
  144. // 结束缓存区
  145. ob_end_flush();
  146. // 安装完成进行跳转
  147. echo '<script>setTimeout(function () { location.href="?s=' . md5('done') . '"; }, 2000)</script>';
  148. exit();
  149. }
  150. exit();
  151. }
  152. // 检测数据库信息
  153. if ($s == 63832) {
  154. $dbhost = $_GET['dbhost'] ?? '';
  155. $dbuser = $_GET['dbuser'] ?? '';
  156. $dbpwd = $_GET['dbpwd'] ?? '';
  157. $dbport = $_GET['dbport'] ?? '';
  158. try {
  159. $dsn = "mysql:host=$dbhost;port={$dbport};charset=utf8";
  160. $pdo = new PDO($dsn, $dbuser, $dbpwd);
  161. echo 'true';
  162. } catch (Exception $e) {
  163. echo 'false';
  164. }
  165. exit();
  166. }
  167. // 安装完成
  168. if ($s == md5('done')) {
  169. require_once(INSTALL_PATH . '/templates/step_4.php');
  170. $fp = fopen(INSTALL_PATH . '/install.lock', 'w');
  171. fwrite($fp, '程序已正确安装,重新安装请删除本文件');
  172. fclose($fp);
  173. exit();
  174. }
  175. // 获取扩展要求数据
  176. function getExtendArray()
  177. {
  178. $data = [
  179. [
  180. 'name' => 'CURL',
  181. 'status' => extension_loaded('curl'),
  182. ],
  183. [
  184. 'name' => 'OpenSSL',
  185. 'status' => extension_loaded('openssl'),
  186. ],
  187. [
  188. 'name' => 'PDO Mysql',
  189. 'status' => extension_loaded('PDO') && extension_loaded('pdo_mysql'),
  190. ],
  191. [
  192. 'name' => 'GD',
  193. 'status' => extension_loaded('gd'),
  194. ],
  195. [
  196. 'name' => 'BCMath',
  197. 'status' => extension_loaded('bcmath'),
  198. ],
  199. [
  200. 'name' => 'mbstring',
  201. 'status' => extension_loaded('mbstring'),
  202. ],
  203. [
  204. 'name' => 'SimpleXML',
  205. 'status' => extension_loaded('SimpleXML'),
  206. ]
  207. ];
  208. foreach ($data as $item) {
  209. !$item['status'] && setIsNext(false);
  210. }
  211. return $data;
  212. }
  213. // 获取检测的路径数据
  214. function getIsWriteArray()
  215. {
  216. return [
  217. '/.env',
  218. // '/data/',
  219. '/public/install/',
  220. '/public/uploads/',
  221. '/public/temp/',
  222. ];
  223. }
  224. // 获取检测的函数数据
  225. function getExistsFuncArray()
  226. {
  227. return ['curl_init', 'bcadd', 'mb_substr', 'simplexml_load_string'];
  228. }
  229. // 测试可写性
  230. function isWrite($file)
  231. {
  232. if (is_writable(ROOT_PATH . $file)) {
  233. echo '<span>可写</span>';
  234. } else {
  235. echo '<span class="col-red">不可写</span>';
  236. setIsNext(false);
  237. }
  238. }
  239. // 测试函数是否存在
  240. function isFunExists($func)
  241. {
  242. $state = function_exists($func);
  243. if ($state === false) {
  244. setIsNext(false);
  245. }
  246. return $state;
  247. }
  248. // 测试函数是否存在
  249. function isFunExistsTxt($func)
  250. {
  251. if (isFunExists($func)) {
  252. echo '<span>无</span>';
  253. } else {
  254. echo '<span class="col-red">需安装</span>';
  255. setIsNext(false);
  256. }
  257. }
  258. // 清除txt中的BOM
  259. function clearBOM($contents)
  260. {
  261. $charset[1] = substr($contents, 0, 1);
  262. $charset[2] = substr($contents, 1, 1);
  263. $charset[3] = substr($contents, 2, 1);
  264. if (ord($charset[1]) == 239 &&
  265. ord($charset[2]) == 187 &&
  266. ord($charset[3]) == 191) {
  267. return substr($contents, 3);
  268. } else {
  269. return $contents;
  270. }
  271. }
  272. // 设置是否允许下一步
  273. function setIsNext(bool $bool)
  274. {
  275. $GLOBALS['isNext'] = $bool;
  276. }
  277. // 获取data文件夹中的文件内容
  278. function readDataFile(string $file)
  279. {
  280. return file_get_contents(INSTALL_PATH . '/data/' . $file);
  281. }
  282. function insInfo($str)
  283. {
  284. echo '<script>$("#install").append("' . $str . '<br>");</script>';
  285. }
  286. function insError($str, $isExit = false)
  287. {
  288. insInfo("<span class='col-red'>$str</span>");
  289. exit();
  290. }