install.php 49 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489
  1. <?php
  2. if (!file_exists(dirname(__FILE__) . '/config.inc.php')) {
  3. // site root path
  4. define('__TYPECHO_ROOT_DIR__', dirname(__FILE__));
  5. // plugin directory (relative path)
  6. define('__TYPECHO_PLUGIN_DIR__', '/usr/plugins');
  7. // theme directory (relative path)
  8. define('__TYPECHO_THEME_DIR__', '/usr/themes');
  9. // admin directory (relative path)
  10. define('__TYPECHO_ADMIN_DIR__', '/admin/');
  11. // register autoload
  12. require_once __TYPECHO_ROOT_DIR__ . '/var/Typecho/Common.php';
  13. // init
  14. \Typecho\Common::init();
  15. } else {
  16. require_once dirname(__FILE__) . '/config.inc.php';
  17. $installDb = \Typecho\Db::get();
  18. }
  19. /**
  20. * get lang
  21. *
  22. * @return string
  23. */
  24. function install_get_lang(): string
  25. {
  26. $serverLang = \Typecho\Request::getInstance()->getServer('TYPECHO_LANG');
  27. if (!empty($serverLang)) {
  28. return $serverLang;
  29. } else {
  30. $lang = 'zh_CN';
  31. $request = \Typecho\Request::getInstance();
  32. if ($request->is('lang')) {
  33. $lang = $request->get('lang');
  34. \Typecho\Cookie::set('lang', $lang);
  35. }
  36. return \Typecho\Cookie::get('lang', $lang);
  37. }
  38. }
  39. /**
  40. * get site url
  41. *
  42. * @return string
  43. */
  44. function install_get_site_url(): string
  45. {
  46. $request = \Typecho\Request::getInstance();
  47. return install_is_cli() ? $request->getServer('TYPECHO_SITE_URL', 'http://localhost') : $request->getRequestRoot();
  48. }
  49. /**
  50. * detect cli mode
  51. *
  52. * @return bool
  53. */
  54. function install_is_cli(): bool
  55. {
  56. return \Typecho\Request::getInstance()->isCli();
  57. }
  58. /**
  59. * get default router
  60. *
  61. * @return string[][]
  62. */
  63. function install_get_default_routers(): array
  64. {
  65. return [
  66. 'index' =>
  67. [
  68. 'url' => '/',
  69. 'widget' => '\Widget\Archive',
  70. 'action' => 'render',
  71. ],
  72. 'archive' =>
  73. [
  74. 'url' => '/blog/',
  75. 'widget' => '\Widget\Archive',
  76. 'action' => 'render',
  77. ],
  78. 'do' =>
  79. [
  80. 'url' => '/action/[action:alpha]',
  81. 'widget' => '\Widget\Action',
  82. 'action' => 'action',
  83. ],
  84. 'post' =>
  85. [
  86. 'url' => '/archives/[cid:digital]/',
  87. 'widget' => '\Widget\Archive',
  88. 'action' => 'render',
  89. ],
  90. 'attachment' =>
  91. [
  92. 'url' => '/attachment/[cid:digital]/',
  93. 'widget' => '\Widget\Archive',
  94. 'action' => 'render',
  95. ],
  96. 'category' =>
  97. [
  98. 'url' => '/category/[slug]/',
  99. 'widget' => '\Widget\Archive',
  100. 'action' => 'render',
  101. ],
  102. 'tag' =>
  103. [
  104. 'url' => '/tag/[slug]/',
  105. 'widget' => '\Widget\Archive',
  106. 'action' => 'render',
  107. ],
  108. 'author' =>
  109. [
  110. 'url' => '/author/[uid:digital]/',
  111. 'widget' => '\Widget\Archive',
  112. 'action' => 'render',
  113. ],
  114. 'search' =>
  115. [
  116. 'url' => '/search/[keywords]/',
  117. 'widget' => '\Widget\Archive',
  118. 'action' => 'render',
  119. ],
  120. 'index_page' =>
  121. [
  122. 'url' => '/page/[page:digital]/',
  123. 'widget' => '\Widget\Archive',
  124. 'action' => 'render',
  125. ],
  126. 'archive_page' =>
  127. [
  128. 'url' => '/blog/page/[page:digital]/',
  129. 'widget' => '\Widget\Archive',
  130. 'action' => 'render',
  131. ],
  132. 'category_page' =>
  133. [
  134. 'url' => '/category/[slug]/[page:digital]/',
  135. 'widget' => '\Widget\Archive',
  136. 'action' => 'render',
  137. ],
  138. 'tag_page' =>
  139. [
  140. 'url' => '/tag/[slug]/[page:digital]/',
  141. 'widget' => '\Widget\Archive',
  142. 'action' => 'render',
  143. ],
  144. 'author_page' =>
  145. [
  146. 'url' => '/author/[uid:digital]/[page:digital]/',
  147. 'widget' => '\Widget\Archive',
  148. 'action' => 'render',
  149. ],
  150. 'search_page' =>
  151. [
  152. 'url' => '/search/[keywords]/[page:digital]/',
  153. 'widget' => '\Widget\Archive',
  154. 'action' => 'render',
  155. ],
  156. 'archive_year' =>
  157. [
  158. 'url' => '/[year:digital:4]/',
  159. 'widget' => '\Widget\Archive',
  160. 'action' => 'render',
  161. ],
  162. 'archive_month' =>
  163. [
  164. 'url' => '/[year:digital:4]/[month:digital:2]/',
  165. 'widget' => '\Widget\Archive',
  166. 'action' => 'render',
  167. ],
  168. 'archive_day' =>
  169. [
  170. 'url' => '/[year:digital:4]/[month:digital:2]/[day:digital:2]/',
  171. 'widget' => '\Widget\Archive',
  172. 'action' => 'render',
  173. ],
  174. 'archive_year_page' =>
  175. [
  176. 'url' => '/[year:digital:4]/page/[page:digital]/',
  177. 'widget' => '\Widget\Archive',
  178. 'action' => 'render',
  179. ],
  180. 'archive_month_page' =>
  181. [
  182. 'url' => '/[year:digital:4]/[month:digital:2]/page/[page:digital]/',
  183. 'widget' => '\Widget\Archive',
  184. 'action' => 'render',
  185. ],
  186. 'archive_day_page' =>
  187. [
  188. 'url' => '/[year:digital:4]/[month:digital:2]/[day:digital:2]/page/[page:digital]/',
  189. 'widget' => '\Widget\Archive',
  190. 'action' => 'render',
  191. ],
  192. 'comment_page' =>
  193. [
  194. 'url' => '[permalink:string]/comment-page-[commentPage:digital]',
  195. 'widget' => '\Widget\Archive',
  196. 'action' => 'render',
  197. ],
  198. 'feed' =>
  199. [
  200. 'url' => '/feed[feed:string:0]',
  201. 'widget' => '\Widget\Archive',
  202. 'action' => 'feed',
  203. ],
  204. 'feedback' =>
  205. [
  206. 'url' => '[permalink:string]/[type:alpha]',
  207. 'widget' => '\Widget\Feedback',
  208. 'action' => 'action',
  209. ],
  210. 'page' =>
  211. [
  212. 'url' => '/[slug].html',
  213. 'widget' => '\Widget\Archive',
  214. 'action' => 'render',
  215. ],
  216. ];
  217. }
  218. /**
  219. * list all default options
  220. *
  221. * @return array
  222. */
  223. function install_get_default_options(): array
  224. {
  225. static $options;
  226. if (empty($options)) {
  227. $options = [
  228. 'theme' => 'default',
  229. 'theme:default' => 'a:2:{s:7:"logoUrl";N;s:12:"sidebarBlock";a:5:{i:0;s:15:"ShowRecentPosts";i:1;s:18:"ShowRecentComments";i:2;s:12:"ShowCategory";i:3;s:11:"ShowArchive";i:4;s:9:"ShowOther";}}',
  230. 'timezone' => '28800',
  231. 'lang' => install_get_lang(),
  232. 'charset' => 'UTF-8',
  233. 'contentType' => 'text/html',
  234. 'gzip' => 0,
  235. 'generator' => 'Typecho ' . \Typecho\Common::VERSION,
  236. 'title' => 'Hello World',
  237. 'description' => 'Your description here.',
  238. 'keywords' => 'typecho,php,blog',
  239. 'rewrite' => 0,
  240. 'frontPage' => 'recent',
  241. 'frontArchive' => 0,
  242. 'commentsRequireMail' => 1,
  243. 'commentsWhitelist' => 0,
  244. 'commentsRequireURL' => 0,
  245. 'commentsRequireModeration' => 0,
  246. 'plugins' => 'a:0:{}',
  247. 'commentDateFormat' => 'F jS, Y \a\t h:i a',
  248. 'siteUrl' => install_get_site_url(),
  249. 'defaultCategory' => 1,
  250. 'allowRegister' => 0,
  251. 'defaultAllowComment' => 1,
  252. 'defaultAllowPing' => 1,
  253. 'defaultAllowFeed' => 1,
  254. 'pageSize' => 5,
  255. 'postsListSize' => 10,
  256. 'commentsListSize' => 10,
  257. 'commentsHTMLTagAllowed' => null,
  258. 'postDateFormat' => 'Y-m-d',
  259. 'feedFullText' => 1,
  260. 'editorSize' => 350,
  261. 'autoSave' => 0,
  262. 'markdown' => 1,
  263. 'xmlrpcMarkdown' => 0,
  264. 'commentsMaxNestingLevels' => 5,
  265. 'commentsPostTimeout' => 24 * 3600 * 30,
  266. 'commentsUrlNofollow' => 1,
  267. 'commentsShowUrl' => 1,
  268. 'commentsMarkdown' => 0,
  269. 'commentsPageBreak' => 0,
  270. 'commentsThreaded' => 1,
  271. 'commentsPageSize' => 20,
  272. 'commentsPageDisplay' => 'last',
  273. 'commentsOrder' => 'ASC',
  274. 'commentsCheckReferer' => 1,
  275. 'commentsAutoClose' => 0,
  276. 'commentsPostIntervalEnable' => 1,
  277. 'commentsPostInterval' => 60,
  278. 'commentsShowCommentOnly' => 0,
  279. 'commentsAvatar' => 1,
  280. 'commentsAvatarRating' => 'G',
  281. 'commentsAntiSpam' => 1,
  282. 'routingTable' => serialize(install_get_default_routers()),
  283. 'actionTable' => 'a:0:{}',
  284. 'panelTable' => 'a:0:{}',
  285. 'attachmentTypes' => '@image@',
  286. 'secret' => \Typecho\Common::randString(32, true),
  287. 'installed' => 0,
  288. 'allowXmlRpc' => 2
  289. ];
  290. }
  291. return $options;
  292. }
  293. /**
  294. * get database driver type
  295. *
  296. * @param string $driver
  297. * @return string
  298. */
  299. function install_get_db_type(string $driver): string
  300. {
  301. $parts = explode('_', $driver);
  302. return $driver == 'Mysqli' ? 'Mysql' : array_pop($parts);
  303. }
  304. /**
  305. * list all available database drivers
  306. *
  307. * @return array
  308. */
  309. function install_get_db_drivers(): array
  310. {
  311. $drivers = [];
  312. if (\Typecho\Db\Adapter\Pdo\Mysql::isAvailable()) {
  313. $drivers['Pdo_Mysql'] = _t('Pdo 驱动 Mysql 适配器');
  314. }
  315. if (\Typecho\Db\Adapter\Pdo\SQLite::isAvailable()) {
  316. $drivers['Pdo_SQLite'] = _t('Pdo 驱动 SQLite 适配器');
  317. }
  318. if (\Typecho\Db\Adapter\Pdo\Pgsql::isAvailable()) {
  319. $drivers['Pdo_Pgsql'] = _t('Pdo 驱动 PostgreSql 适配器');
  320. }
  321. if (\Typecho\Db\Adapter\Mysqli::isAvailable()) {
  322. $drivers['Mysqli'] = _t('Mysql 原生函数适配器');
  323. }
  324. if (\Typecho\Db\Adapter\SQLite::isAvailable()) {
  325. $drivers['SQLite'] = _t('SQLite 原生函数适配器');
  326. }
  327. if (\Typecho\Db\Adapter\Pgsql::isAvailable()) {
  328. $drivers['Pgsql'] = _t('Pgsql 原生函数适配器');
  329. }
  330. return $drivers;
  331. }
  332. /**
  333. * get current db driver
  334. *
  335. * @return string
  336. */
  337. function install_get_current_db_driver(): string
  338. {
  339. global $installDb;
  340. if (empty($installDb)) {
  341. $driver = \Typecho\Request::getInstance()->get('driver');
  342. $drivers = install_get_db_drivers();
  343. if (empty($driver) || !isset($drivers[$driver])) {
  344. return key($drivers);
  345. }
  346. return $driver;
  347. } else {
  348. return $installDb->getAdapterName();
  349. }
  350. }
  351. /**
  352. * generate config file
  353. *
  354. * @param string $adapter
  355. * @param string $dbPrefix
  356. * @param array $dbConfig
  357. * @param bool $return
  358. * @return string
  359. */
  360. function install_config_file(string $adapter, string $dbPrefix, array $dbConfig, bool $return = false): string
  361. {
  362. global $configWritten;
  363. $code = "<" . "?php
  364. // site root path
  365. define('__TYPECHO_ROOT_DIR__', dirname(__FILE__));
  366. // plugin directory (relative path)
  367. define('__TYPECHO_PLUGIN_DIR__', '/usr/plugins');
  368. // theme directory (relative path)
  369. define('__TYPECHO_THEME_DIR__', '/usr/themes');
  370. // admin directory (relative path)
  371. define('__TYPECHO_ADMIN_DIR__', '/admin/');
  372. // register autoload
  373. require_once __TYPECHO_ROOT_DIR__ . '/var/Typecho/Common.php';
  374. // init
  375. \Typecho\Common::init();
  376. // config db
  377. \$db = new \Typecho\Db('{$adapter}', '{$dbPrefix}');
  378. \$db->addServer(" . (var_export($dbConfig, true)) . ", \Typecho\Db::READ | \Typecho\Db::WRITE);
  379. \Typecho\Db::set(\$db);
  380. ";
  381. $configWritten = false;
  382. if (!$return) {
  383. $configWritten = @file_put_contents(__TYPECHO_ROOT_DIR__ . '/config.inc.php', $code) !== false;
  384. }
  385. return $code;
  386. }
  387. /**
  388. * remove config file if written
  389. */
  390. function install_remove_config_file()
  391. {
  392. global $configWritten;
  393. if ($configWritten) {
  394. unlink(__TYPECHO_ROOT_DIR__ . '/config.inc.php');
  395. }
  396. }
  397. /**
  398. * check install
  399. *
  400. * @param string $type
  401. * @return bool
  402. */
  403. function install_check(string $type): bool
  404. {
  405. switch ($type) {
  406. case 'config':
  407. return file_exists(__TYPECHO_ROOT_DIR__ . '/config.inc.php');
  408. case 'db_structure':
  409. case 'db_data':
  410. global $installDb;
  411. if (empty($installDb)) {
  412. return false;
  413. }
  414. try {
  415. // check if table exists
  416. $installed = $installDb->fetchRow($installDb->select()->from('table.options')
  417. ->where('user = 0 AND name = ?', 'installed'));
  418. if ($type == 'db_data' && empty($installed['value'])) {
  419. return false;
  420. }
  421. } catch (\Typecho\Db\Adapter\ConnectionException $e) {
  422. return true;
  423. } catch (\Typecho\Db\Adapter\SQLException $e) {
  424. return false;
  425. }
  426. return true;
  427. default:
  428. return false;
  429. }
  430. }
  431. /**
  432. * raise install error
  433. *
  434. * @param mixed $error
  435. * @param mixed $config
  436. */
  437. function install_raise_error($error, $config = null)
  438. {
  439. if (install_is_cli()) {
  440. if (is_array($error)) {
  441. foreach ($error as $key => $value) {
  442. echo (is_int($key) ? '' : $key . ': ') . $value . "\n";
  443. }
  444. } else {
  445. echo $error . "\n";
  446. }
  447. exit(1);
  448. } else {
  449. install_throw_json([
  450. 'success' => 0,
  451. 'message' => is_string($error) ? nl2br($error) : $error,
  452. 'config' => $config
  453. ]);
  454. }
  455. }
  456. /**
  457. * @param $step
  458. * @param array|null $config
  459. */
  460. function install_success($step, ?array $config = null)
  461. {
  462. global $installDb;
  463. if (install_is_cli()) {
  464. if ($step == 3) {
  465. \Typecho\Db::set($installDb);
  466. }
  467. if ($step > 0) {
  468. $method = 'install_step_' . $step . '_perform';
  469. $method();
  470. }
  471. if (!empty($config)) {
  472. [$userName, $userPassword] = $config;
  473. echo _t('安装成功') . "\n";
  474. echo _t('您的用户名是') . " {$userName}\n";
  475. echo _t('您的密码是') . " {$userPassword}\n";
  476. }
  477. exit(0);
  478. } else {
  479. install_throw_json([
  480. 'success' => 1,
  481. 'message' => $step,
  482. 'config' => $config
  483. ]);
  484. }
  485. }
  486. /**
  487. * @param $data
  488. */
  489. function install_throw_json($data)
  490. {
  491. \Typecho\Response::getInstance()->setContentType('application/json')
  492. ->addResponder(function () use ($data) {
  493. echo json_encode($data);
  494. })
  495. ->respond();
  496. }
  497. /**
  498. * @param string $url
  499. */
  500. function install_redirect(string $url)
  501. {
  502. \Typecho\Response::getInstance()->setStatus(302)
  503. ->setHeader('Location', $url)
  504. ->respond();
  505. }
  506. /**
  507. * add common js support
  508. */
  509. function install_js_support()
  510. {
  511. ?>
  512. <div id="success" class="row typecho-page-main hidden">
  513. <div class="col-mb-12 col-tb-8 col-tb-offset-2">
  514. <div class="typecho-page-title">
  515. <h2><?php _e('安装成功'); ?></h2>
  516. </div>
  517. <div id="typecho-welcome">
  518. <p class="keep-word">
  519. <?php _e('您选择了使用原有的数据, 您的用户名和密码和原来的一致'); ?>
  520. </p>
  521. <p class="fresh-word">
  522. <?php _e('您的用户名是'); ?>: <strong class="warning" id="success-user"></strong><br>
  523. <?php _e('您的密码是'); ?>: <strong class="warning" id="success-password"></strong>
  524. </p>
  525. <ul>
  526. <li><a id="login-url" href=""><?php _e('点击这里访问您的控制面板'); ?></a></li>
  527. <li><a id="site-url" href=""><?php _e('点击这里查看您的 Blog'); ?></a></li>
  528. </ul>
  529. <p><?php _e('希望您能尽情享用 Typecho 带来的乐趣!'); ?></p>
  530. </div>
  531. </div>
  532. </div>
  533. <script>
  534. let form = $('form'), errorBox = $('<div></div>');
  535. errorBox.addClass('message error')
  536. .prependTo(form);
  537. function showError(error) {
  538. if (typeof error == 'string') {
  539. $(window).scrollTop(0);
  540. errorBox
  541. .html(error)
  542. .addClass('fade');
  543. } else {
  544. for (let k in error) {
  545. let input = $('#' + k), msg = error[k], p = $('<p></p>');
  546. p.addClass('message error')
  547. .html(msg)
  548. .insertAfter(input);
  549. input.on('input', function () {
  550. p.remove();
  551. });
  552. }
  553. }
  554. return errorBox;
  555. }
  556. form.submit(function (e) {
  557. e.preventDefault();
  558. errorBox.removeClass('fade');
  559. $('button', form).attr('disabled', 'disabled');
  560. $('.typecho-option .error', form).remove();
  561. $.ajax({
  562. url: form.attr('action'),
  563. processData: false,
  564. contentType: false,
  565. type: 'POST',
  566. data: new FormData(this),
  567. success: function (data) {
  568. $('button', form).removeAttr('disabled');
  569. if (data.success) {
  570. if (data.message) {
  571. location.href = '?step=' + data.message;
  572. } else {
  573. let success = $('#success').removeClass('hidden');
  574. form.addClass('hidden');
  575. if (data.config) {
  576. success.addClass('fresh');
  577. $('.typecho-page-main:first').addClass('hidden');
  578. $('#success-user').html(data.config[0]);
  579. $('#success-password').html(data.config[1]);
  580. $('#login-url').attr('href', data.config[2]);
  581. $('#site-url').attr('href', data.config[3]);
  582. } else {
  583. success.addClass('keep');
  584. }
  585. }
  586. } else {
  587. let el = showError(data.message);
  588. if (typeof configError == 'function' && data.config) {
  589. configError(form, data.config, el);
  590. }
  591. }
  592. },
  593. error: function (xhr, error) {
  594. showError(error)
  595. }
  596. });
  597. });
  598. </script>
  599. <?php
  600. }
  601. /**
  602. * @param string[] $extensions
  603. * @return string|null
  604. */
  605. function install_check_extension(array $extensions): ?string
  606. {
  607. foreach ($extensions as $extension) {
  608. if (extension_loaded($extension)) {
  609. return null;
  610. }
  611. }
  612. return _n('缺少PHP扩展', '请在服务器上安装以下PHP扩展中的至少一个', count($extensions))
  613. . ': ' . implode(', ', $extensions);
  614. }
  615. function install_step_1()
  616. {
  617. $langs = \Widget\Options\General::getLangs();
  618. $lang = install_get_lang();
  619. ?>
  620. <div class="row typecho-page-main">
  621. <div class="col-mb-12 col-tb-8 col-tb-offset-2">
  622. <div class="typecho-page-title">
  623. <h2><?php _e('欢迎使用 Typecho'); ?></h2>
  624. </div>
  625. <div id="typecho-welcome">
  626. <form autocomplete="off" method="post" action="install.php">
  627. <h3><?php _e('安装说明'); ?></h3>
  628. <p class="warning">
  629. <strong><?php _e('本安装程序将自动检测服务器环境是否符合最低配置需求. 如果不符合, 将在上方出现提示信息, 请按照提示信息检查您的主机配置. 如果服务器环境符合要求, 将在下方出现 "开始下一步" 的按钮, 点击此按钮即可一步完成安装.'); ?></strong>
  630. </p>
  631. <h3><?php _e('许可及协议'); ?></h3>
  632. <ul>
  633. <li><?php _e('Typecho 基于 <a href="https://www.gnu.org/copyleft/gpl.html">GPL</a> 协议发布, 我们允许用户在 GPL 协议许可的范围内使用, 拷贝, 修改和分发此程序.'); ?>
  634. <?php _e('在GPL许可的范围内, 您可以自由地将其用于商业以及非商业用途.'); ?></li>
  635. <li><?php _e('Typecho 软件由其社区提供支持, 核心开发团队负责维护程序日常开发工作以及新特性的制定.'); ?>
  636. <?php _e('如果您遇到使用上的问题, 程序中的 BUG, 以及期许的新功能, 欢迎您在社区中交流或者直接向我们贡献代码.'); ?>
  637. <?php _e('对于贡献突出者, 他的名字将出现在贡献者名单中.'); ?></li>
  638. </ul>
  639. <p class="submit">
  640. <button class="btn primary" type="submit"><?php _e('我准备好了, 开始下一步 &raquo;'); ?></button>
  641. <input type="hidden" name="step" value="1">
  642. <?php if (count($langs) > 1) : ?>
  643. <select style="float: right" onchange="location.href='?lang=' + this.value">
  644. <?php foreach ($langs as $key => $val) : ?>
  645. <option value="<?php echo $key; ?>"<?php if ($lang == $key) :
  646. ?> selected<?php
  647. endif; ?>><?php echo $val; ?></option>
  648. <?php endforeach; ?>
  649. </select>
  650. <?php endif; ?>
  651. </p>
  652. </form>
  653. </div>
  654. </div>
  655. </div>
  656. <?php
  657. install_js_support();
  658. }
  659. /**
  660. * check dependencies before install
  661. */
  662. function install_step_1_perform()
  663. {
  664. $errors = [];
  665. $checks = [
  666. 'mbstring',
  667. 'json',
  668. 'Reflection',
  669. ['mysqli', 'sqlite3', 'pgsql', 'pdo_mysql', 'pdo_sqlite', 'pdo_pgsql']
  670. ];
  671. foreach ($checks as $check) {
  672. $error = install_check_extension(is_array($check) ? $check : [$check]);
  673. if (!empty($error)) {
  674. $errors[] = $error;
  675. }
  676. }
  677. $uploadDir = '/usr/uploads';
  678. $realUploadDir = \Typecho\Common::url($uploadDir, __TYPECHO_ROOT_DIR__);
  679. $writeable = true;
  680. if (is_dir($realUploadDir)) {
  681. if (!is_writeable($realUploadDir) || !is_readable($realUploadDir)) {
  682. if (!@chmod($realUploadDir, 0755)) {
  683. $writeable = false;
  684. }
  685. }
  686. } else {
  687. if (!@mkdir($realUploadDir, 0755)) {
  688. $writeable = false;
  689. }
  690. }
  691. if (!$writeable) {
  692. $errors[] = _t('上传目录无法写入, 请手动将安装目录下的 %s 目录的权限设置为可写然后继续升级', $uploadDir);
  693. }
  694. if (empty($errors)) {
  695. install_success(2);
  696. } else {
  697. install_raise_error(implode("\n", $errors));
  698. }
  699. }
  700. /**
  701. * display step 2
  702. */
  703. function install_step_2()
  704. {
  705. global $installDb;
  706. $drivers = install_get_db_drivers();
  707. $adapter = install_get_current_db_driver();
  708. $type = install_get_db_type($adapter);
  709. if (!empty($installDb)) {
  710. $config = $installDb->getConfig(\Typecho\Db::WRITE)->toArray();
  711. $config['prefix'] = $installDb->getPrefix();
  712. $config['adapter'] = $adapter;
  713. }
  714. ?>
  715. <div class="row typecho-page-main">
  716. <div class="col-mb-12 col-tb-8 col-tb-offset-2">
  717. <div class="typecho-page-title">
  718. <h2><?php _e('初始化配置'); ?></h2>
  719. </div>
  720. <form autocomplete="off" action="install.php" method="post">
  721. <ul class="typecho-option">
  722. <li>
  723. <label for="dbAdapter" class="typecho-label"><?php _e('数据库适配器'); ?></label>
  724. <select name="dbAdapter" id="dbAdapter" onchange="location.href='?step=2&driver=' + this.value">
  725. <?php foreach ($drivers as $driver => $name) : ?>
  726. <option value="<?php echo $driver; ?>"<?php if ($driver == $adapter) :
  727. ?> selected="selected"<?php
  728. endif; ?>><?php echo $name; ?></option>
  729. <?php endforeach; ?>
  730. </select>
  731. <p class="description"><?php _e('请根据您的数据库类型选择合适的适配器'); ?></p>
  732. <input type="hidden" id="dbNext" name="dbNext" value="none">
  733. </li>
  734. </ul>
  735. <ul class="typecho-option">
  736. <li>
  737. <label class="typecho-label" for="dbPrefix"><?php _e('数据库前缀'); ?></label>
  738. <input type="text" class="text" name="dbPrefix" id="dbPrefix" value="typecho_" />
  739. <p class="description"><?php _e('默认前缀是 "typecho_"'); ?></p>
  740. </li>
  741. </ul>
  742. <?php require_once './install/' . $type . '.php'; ?>
  743. <ul class="typecho-option typecho-option-submit">
  744. <li>
  745. <button id="confirm" type="submit" class="btn primary"><?php _e('确认, 开始安装 &raquo;'); ?></button>
  746. <input type="hidden" name="step" value="2">
  747. </li>
  748. </ul>
  749. </form>
  750. </div>
  751. </div>
  752. <script>
  753. function configError(form, config, errorBox) {
  754. let next = $('#dbNext'),
  755. line = $('<p></p>');
  756. if (config.code) {
  757. let text = $('<textarea></textarea>'),
  758. btn = $('<button></button>');
  759. btn.html('<?php _e('创建完毕, 继续安装 &raquo;'); ?>')
  760. .attr('type', 'button')
  761. .addClass('btn btn-s primary');
  762. btn.click(function () {
  763. next.val('config');
  764. form.trigger('submit');
  765. });
  766. text.val(config.code)
  767. .addClass('mono')
  768. .attr('readonly', 'readonly');
  769. errorBox.append(text)
  770. .append(btn);
  771. return;
  772. }
  773. errorBox.append(line);
  774. for (let key in config) {
  775. let word = config[key],
  776. btn = $('<button></button>');
  777. btn.html(word)
  778. .attr('type', 'button')
  779. .addClass('btn btn-s primary')
  780. .click(function () {
  781. next.val(key);
  782. form.trigger('submit');
  783. });
  784. line.append(btn);
  785. }
  786. }
  787. $('#confirm').click(function () {
  788. $('#dbNext').val('none');
  789. });
  790. <?php if (!empty($config)) : ?>
  791. function fillInput(config) {
  792. for (let k in config) {
  793. let value = config[k],
  794. key = 'db' + k.charAt(0).toUpperCase() + k.slice(1),
  795. input = $('#' + key)
  796. .attr('readonly', 'readonly')
  797. .val(value);
  798. $('option:not(:selected)', input).attr('disabled', 'disabled');
  799. }
  800. }
  801. fillInput(<?php echo json_encode($config); ?>);
  802. <?php endif; ?>
  803. </script>
  804. <?php
  805. install_js_support();
  806. }
  807. /**
  808. * perform install step 2
  809. */
  810. function install_step_2_perform()
  811. {
  812. global $installDb;
  813. $request = \Typecho\Request::getInstance();
  814. $drivers = install_get_db_drivers();
  815. $configMap = [
  816. 'Mysql' => [
  817. 'dbHost' => 'localhost',
  818. 'dbPort' => 3306,
  819. 'dbUser' => null,
  820. 'dbPassword' => null,
  821. 'dbCharset' => 'utf8mb4',
  822. 'dbDatabase' => null,
  823. 'dbEngine' => 'InnoDB',
  824. 'dbSslCa' => null,
  825. 'dbSslVerify' => 'on',
  826. ],
  827. 'Pgsql' => [
  828. 'dbHost' => 'localhost',
  829. 'dbPort' => 5432,
  830. 'dbUser' => null,
  831. 'dbPassword' => null,
  832. 'dbCharset' => 'utf8',
  833. 'dbDatabase' => null,
  834. ],
  835. 'SQLite' => [
  836. 'dbFile' => __TYPECHO_ROOT_DIR__ . '/usr/' . uniqid() . '.db'
  837. ]
  838. ];
  839. if (install_is_cli()) {
  840. $config = [
  841. 'dbHost' => $request->getServer('TYPECHO_DB_HOST'),
  842. 'dbUser' => $request->getServer('TYPECHO_DB_USER'),
  843. 'dbPassword' => $request->getServer('TYPECHO_DB_PASSWORD'),
  844. 'dbCharset' => $request->getServer('TYPECHO_DB_CHARSET'),
  845. 'dbPort' => $request->getServer('TYPECHO_DB_PORT'),
  846. 'dbDatabase' => $request->getServer('TYPECHO_DB_DATABASE'),
  847. 'dbFile' => $request->getServer('TYPECHO_DB_FILE'),
  848. 'dbDsn' => $request->getServer('TYPECHO_DB_DSN'),
  849. 'dbEngine' => $request->getServer('TYPECHO_DB_ENGINE'),
  850. 'dbPrefix' => $request->getServer('TYPECHO_DB_PREFIX', 'typecho_'),
  851. 'dbAdapter' => $request->getServer('TYPECHO_DB_ADAPTER', install_get_current_db_driver()),
  852. 'dbNext' => $request->getServer('TYPECHO_DB_NEXT', 'none'),
  853. 'dbSslCa' => $request->getServer('TYPECHO_DB_SSL_CA'),
  854. 'dbSslVerify' => $request->getServer('TYPECHO_DB_SSL_VERIFY', 'on'),
  855. ];
  856. } else {
  857. $config = $request->from([
  858. 'dbHost',
  859. 'dbUser',
  860. 'dbPassword',
  861. 'dbCharset',
  862. 'dbPort',
  863. 'dbDatabase',
  864. 'dbFile',
  865. 'dbDsn',
  866. 'dbEngine',
  867. 'dbPrefix',
  868. 'dbAdapter',
  869. 'dbNext',
  870. 'dbSslCa',
  871. 'dbSslVerify',
  872. ]);
  873. }
  874. $error = (new \Typecho\Validate())
  875. ->addRule('dbPrefix', 'required', _t('确认您的配置'))
  876. ->addRule('dbPrefix', 'minLength', _t('确认您的配置'), 1)
  877. ->addRule('dbPrefix', 'maxLength', _t('确认您的配置'), 16)
  878. ->addRule('dbPrefix', 'alphaDash', _t('确认您的配置'))
  879. ->addRule('dbAdapter', 'required', _t('确认您的配置'))
  880. ->addRule('dbAdapter', 'enum', _t('确认您的配置'), array_keys($drivers))
  881. ->addRule('dbNext', 'required', _t('确认您的配置'))
  882. ->addRule('dbNext', 'enum', _t('确认您的配置'), ['none', 'delete', 'keep', 'config'])
  883. ->run($config);
  884. if (!empty($error)) {
  885. install_raise_error($error);
  886. }
  887. $type = install_get_db_type($config['dbAdapter']);
  888. $dbConfig = [];
  889. foreach ($configMap[$type] as $key => $value) {
  890. $config[$key] = !isset($config[$key]) ? (install_is_cli() ? $value : null) : $config[$key];
  891. }
  892. switch ($type) {
  893. case 'Mysql':
  894. $error = (new \Typecho\Validate())
  895. ->addRule('dbHost', 'required', _t('确认您的配置'))
  896. ->addRule('dbPort', 'required', _t('确认您的配置'))
  897. ->addRule('dbPort', 'isInteger', _t('确认您的配置'))
  898. ->addRule('dbUser', 'required', _t('确认您的配置'))
  899. ->addRule('dbCharset', 'required', _t('确认您的配置'))
  900. ->addRule('dbCharset', 'enum', _t('确认您的配置'), ['utf8', 'utf8mb4'])
  901. ->addRule('dbDatabase', 'required', _t('确认您的配置'))
  902. ->addRule('dbEngine', 'required', _t('确认您的配置'))
  903. ->addRule('dbEngine', 'enum', _t('确认您的配置'), ['InnoDB', 'MyISAM'])
  904. ->addRule('dbSslCa', 'file_exists', _t('确认您的配置'))
  905. ->addRule('dbSslVerify', 'enum', _t('确认您的配置'), ['on', 'off'])
  906. ->run($config);
  907. break;
  908. case 'Pgsql':
  909. $error = (new \Typecho\Validate())
  910. ->addRule('dbHost', 'required', _t('确认您的配置'))
  911. ->addRule('dbPort', 'required', _t('确认您的配置'))
  912. ->addRule('dbPort', 'isInteger', _t('确认您的配置'))
  913. ->addRule('dbUser', 'required', _t('确认您的配置'))
  914. ->addRule('dbCharset', 'required', _t('确认您的配置'))
  915. ->addRule('dbCharset', 'enum', _t('确认您的配置'), ['utf8'])
  916. ->addRule('dbDatabase', 'required', _t('确认您的配置'))
  917. ->run($config);
  918. break;
  919. case 'SQLite':
  920. $error = (new \Typecho\Validate())
  921. ->addRule('dbFile', 'required', _t('确认您的配置'))
  922. ->addRule('dbFile', function (string $path) {
  923. $pattern = "/^(\/[._a-z0-9-]+)*[a-z0-9]+\.[a-z0-9]{2,}$/i";
  924. if (strstr(PHP_OS, 'WIN')) {
  925. $pattern = "/(\/[._a-z0-9-]+)*[a-z0-9]+\.[a-z0-9]{2,}$/i";
  926. }
  927. return !!preg_match($pattern, $path);
  928. }, _t('确认您的配置'))
  929. ->run($config);
  930. break;
  931. default:
  932. install_raise_error(_t('确认您的配置'));
  933. break;
  934. }
  935. if (!empty($error)) {
  936. install_raise_error($error);
  937. }
  938. foreach ($configMap[$type] as $key => $value) {
  939. $dbConfig[lcfirst(substr($key, 2))] = $config[$key];
  940. }
  941. // intval port number
  942. if (isset($dbConfig['port'])) {
  943. $dbConfig['port'] = intval($dbConfig['port']);
  944. }
  945. // bool ssl verify
  946. if (isset($dbConfig['sslVerify'])) {
  947. $dbConfig['sslVerify'] = $dbConfig['sslVerify'] == 'on';
  948. }
  949. if (isset($dbConfig['file']) && preg_match("/^[a-z0-9]+\.[a-z0-9]{2,}$/i", $dbConfig['file'])) {
  950. $dbConfig['file'] = __DIR__ . '/usr/' . $dbConfig['file'];
  951. }
  952. // check config file
  953. if ($config['dbNext'] == 'config' && !install_check('config')) {
  954. $code = install_config_file($config['dbAdapter'], $config['dbPrefix'], $dbConfig, true);
  955. install_raise_error(_t('没有检测到您手动创建的配置文件, 请检查后再次创建'), ['code' => $code]);
  956. } elseif (empty($installDb)) {
  957. // detect db config
  958. try {
  959. $installDb = new \Typecho\Db($config['dbAdapter'], $config['dbPrefix']);
  960. $installDb->addServer($dbConfig, \Typecho\Db::READ | \Typecho\Db::WRITE);
  961. $installDb->query('SELECT 1=1');
  962. } catch (\Typecho\Db\Adapter\ConnectionException $e) {
  963. $code = $e->getCode();
  964. if (('Mysql' == $type && 1049 == $code) || ('Pgsql' == $type && 7 == $code)) {
  965. install_raise_error(_t('数据库: "%s"不存在,请手动创建后重试', $config['dbDatabase']));
  966. } else {
  967. install_raise_error(_t('对不起, 无法连接数据库, 请先检查数据库配置再继续进行安装: "%s"', $e->getMessage()));
  968. }
  969. } catch (\Typecho\Db\Exception $e) {
  970. install_raise_error(_t('安装程序捕捉到以下错误: "%s". 程序被终止, 请检查您的配置信息.', $e->getMessage()));
  971. }
  972. $code = install_config_file($config['dbAdapter'], $config['dbPrefix'], $dbConfig);
  973. if (!install_check('config')) {
  974. install_raise_error(
  975. _t('安装程序无法自动创建 <strong>config.inc.php</strong> 文件') . "\n" .
  976. _t('您可以在网站根目录下手动创建 <strong>config.inc.php</strong> 文件, 并复制如下代码至其中'),
  977. [
  978. 'code' => $code
  979. ]
  980. );
  981. }
  982. }
  983. // delete exists db
  984. if ($config['dbNext'] == 'delete') {
  985. $tables = [
  986. $config['dbPrefix'] . 'comments',
  987. $config['dbPrefix'] . 'contents',
  988. $config['dbPrefix'] . 'fields',
  989. $config['dbPrefix'] . 'metas',
  990. $config['dbPrefix'] . 'options',
  991. $config['dbPrefix'] . 'relationships',
  992. $config['dbPrefix'] . 'users'
  993. ];
  994. try {
  995. foreach ($tables as $table) {
  996. switch ($type) {
  997. case 'Mysql':
  998. $installDb->query("DROP TABLE IF EXISTS `{$table}`");
  999. break;
  1000. case 'Pgsql':
  1001. case 'SQLite':
  1002. $installDb->query("DROP TABLE {$table}");
  1003. break;
  1004. }
  1005. }
  1006. } catch (\Typecho\Db\Exception $e) {
  1007. install_raise_error(_t('安装程序捕捉到以下错误: "%s". 程序被终止, 请检查您的配置信息.', $e->getMessage()));
  1008. }
  1009. }
  1010. // init db structure
  1011. try {
  1012. $scripts = file_get_contents(__TYPECHO_ROOT_DIR__ . '/install/' . $type . '.sql');
  1013. $scripts = str_replace('typecho_', $config['dbPrefix'], $scripts);
  1014. if (isset($dbConfig['charset'])) {
  1015. $scripts = str_replace('%charset%', $dbConfig['charset'], $scripts);
  1016. }
  1017. if (isset($dbConfig['engine'])) {
  1018. $scripts = str_replace('%engine%', $dbConfig['engine'], $scripts);
  1019. }
  1020. $scripts = explode(';', $scripts);
  1021. foreach ($scripts as $script) {
  1022. $script = trim($script);
  1023. if ($script) {
  1024. $installDb->query($script, \Typecho\Db::WRITE);
  1025. }
  1026. }
  1027. } catch (\Typecho\Db\Exception $e) {
  1028. $code = $e->getCode();
  1029. if (
  1030. ('Mysql' == $type && (1050 == $code || '42S01' == $code)) ||
  1031. ('SQLite' == $type && ('HY000' == $code || 1 == $code)) ||
  1032. ('Pgsql' == $type && '42P07' == $code)
  1033. ) {
  1034. if ($config['dbNext'] == 'keep') {
  1035. if (install_check('db_data')) {
  1036. install_success(0);
  1037. } else {
  1038. install_success(3);
  1039. }
  1040. } elseif ($config['dbNext'] == 'none') {
  1041. install_remove_config_file();
  1042. install_raise_error(_t('安装程序检查到原有数据表已经存在.'), [
  1043. 'delete' => _t('删除原有数据'),
  1044. 'keep' => _t('使用原有数据')
  1045. ]);
  1046. }
  1047. } else {
  1048. install_remove_config_file();
  1049. install_raise_error(_t('安装程序捕捉到以下错误: "%s". 程序被终止, 请检查您的配置信息.', $e->getMessage()));
  1050. }
  1051. }
  1052. install_success(3);
  1053. }
  1054. /**
  1055. * display step 3
  1056. */
  1057. function install_step_3()
  1058. {
  1059. $options = \Widget\Options::alloc();
  1060. ?>
  1061. <div class="row typecho-page-main">
  1062. <div class="col-mb-12 col-tb-8 col-tb-offset-2">
  1063. <div class="typecho-page-title">
  1064. <h2><?php _e('创建您的管理员帐号'); ?></h2>
  1065. </div>
  1066. <form autocomplete="off" action="install.php" method="post">
  1067. <ul class="typecho-option">
  1068. <li>
  1069. <label class="typecho-label" for="userUrl"><?php _e('网站地址'); ?></label>
  1070. <input autocomplete="new-password" type="text" name="userUrl" id="userUrl" class="text" value="<?php $options->rootUrl(); ?>" />
  1071. <p class="description"><?php _e('这是程序自动匹配的网站路径, 如果不正确请修改它'); ?></p>
  1072. </li>
  1073. </ul>
  1074. <ul class="typecho-option">
  1075. <li>
  1076. <label class="typecho-label" for="userName"><?php _e('用户名'); ?></label>
  1077. <input autocomplete="new-password" type="text" name="userName" id="userName" class="text" />
  1078. <p class="description"><?php _e('请填写您的用户名'); ?></p>
  1079. </li>
  1080. </ul>
  1081. <ul class="typecho-option">
  1082. <li>
  1083. <label class="typecho-label" for="userPassword"><?php _e('登录密码'); ?></label>
  1084. <input type="password" name="userPassword" id="userPassword" class="text" />
  1085. <p class="description"><?php _e('请填写您的登录密码, 如果留空系统将为您随机生成一个'); ?></p>
  1086. </li>
  1087. </ul>
  1088. <ul class="typecho-option">
  1089. <li>
  1090. <label class="typecho-label" for="userMail"><?php _e('邮件地址'); ?></label>
  1091. <input autocomplete="new-password" type="text" name="userMail" id="userMail" class="text" />
  1092. <p class="description"><?php _e('请填写一个您的常用邮箱'); ?></p>
  1093. </li>
  1094. </ul>
  1095. <ul class="typecho-option typecho-option-submit">
  1096. <li>
  1097. <button type="submit" class="btn primary"><?php _e('继续安装 &raquo;'); ?></button>
  1098. <input type="hidden" name="step" value="3">
  1099. </li>
  1100. </ul>
  1101. </form>
  1102. </div>
  1103. </div>
  1104. <?php
  1105. install_js_support();
  1106. }
  1107. /**
  1108. * perform step 3
  1109. */
  1110. function install_step_3_perform()
  1111. {
  1112. global $installDb;
  1113. $request = \Typecho\Request::getInstance();
  1114. $defaultPassword = \Typecho\Common::randString(8);
  1115. $options = \Widget\Options::alloc();
  1116. if (install_is_cli()) {
  1117. $config = [
  1118. 'userUrl' => $request->getServer('TYPECHO_SITE_URL'),
  1119. 'userName' => $request->getServer('TYPECHO_USER_NAME', 'typecho'),
  1120. 'userPassword' => $request->getServer('TYPECHO_USER_PASSWORD'),
  1121. 'userMail' => $request->getServer('TYPECHO_USER_MAIL', 'admin@localhost.local')
  1122. ];
  1123. } else {
  1124. $config = $request->from([
  1125. 'userUrl',
  1126. 'userName',
  1127. 'userPassword',
  1128. 'userMail',
  1129. ]);
  1130. }
  1131. $error = (new \Typecho\Validate())
  1132. ->addRule('userUrl', 'required', _t('请填写站点地址'))
  1133. ->addRule('userUrl', 'url', _t('请填写一个合法的URL地址'))
  1134. ->addRule('userName', 'required', _t('必须填写用户名称'))
  1135. ->addRule('userName', 'xssCheck', _t('请不要在用户名中使用特殊字符'))
  1136. ->addRule('userName', 'maxLength', _t('用户名长度超过限制, 请不要超过 32 个字符'), 32)
  1137. ->addRule('userMail', 'required', _t('必须填写电子邮箱'))
  1138. ->addRule('userMail', 'email', _t('电子邮箱格式错误'))
  1139. ->addRule('userMail', 'maxLength', _t('邮箱长度超过限制, 请不要超过 200 个字符'), 200)
  1140. ->run($config);
  1141. if (!empty($error)) {
  1142. install_raise_error($error);
  1143. }
  1144. if (empty($config['userPassword'])) {
  1145. $config['userPassword'] = $defaultPassword;
  1146. }
  1147. try {
  1148. // write user
  1149. $hasher = new \Utils\PasswordHash(8, true);
  1150. $installDb->query(
  1151. $installDb->insert('table.users')->rows([
  1152. 'name' => $config['userName'],
  1153. 'password' => $hasher->hashPassword($config['userPassword']),
  1154. 'mail' => $config['userMail'],
  1155. 'url' => $config['userUrl'],
  1156. 'screenName' => $config['userName'],
  1157. 'group' => 'administrator',
  1158. 'created' => \Typecho\Date::time()
  1159. ])
  1160. );
  1161. // write category
  1162. $installDb->query(
  1163. $installDb->insert('table.metas')
  1164. ->rows([
  1165. 'name' => _t('默认分类'),
  1166. 'slug' => 'default',
  1167. 'type' => 'category',
  1168. 'description' => _t('只是一个默认分类'),
  1169. 'count' => 1
  1170. ])
  1171. );
  1172. $installDb->query($installDb->insert('table.relationships')->rows(['cid' => 1, 'mid' => 1]));
  1173. // write first page and post
  1174. $installDb->query(
  1175. $installDb->insert('table.contents')->rows([
  1176. 'title' => _t('欢迎使用 Typecho'),
  1177. 'slug' => 'start', 'created' => \Typecho\Date::time(),
  1178. 'modified' => \Typecho\Date::time(),
  1179. 'text' => '<!--markdown-->' . _t('如果您看到这篇文章,表示您的 blog 已经安装成功.'),
  1180. 'authorId' => 1,
  1181. 'type' => 'post',
  1182. 'status' => 'publish',
  1183. 'commentsNum' => 1,
  1184. 'allowComment' => 1,
  1185. 'allowPing' => 1,
  1186. 'allowFeed' => 1,
  1187. 'parent' => 0
  1188. ])
  1189. );
  1190. $installDb->query(
  1191. $installDb->insert('table.contents')->rows([
  1192. 'title' => _t('关于'),
  1193. 'slug' => 'start-page',
  1194. 'created' => \Typecho\Date::time(),
  1195. 'modified' => \Typecho\Date::time(),
  1196. 'text' => '<!--markdown-->' . _t('本页面由 Typecho 创建, 这只是个测试页面.'),
  1197. 'authorId' => 1,
  1198. 'order' => 0,
  1199. 'type' => 'page',
  1200. 'status' => 'publish',
  1201. 'commentsNum' => 0,
  1202. 'allowComment' => 1,
  1203. 'allowPing' => 1,
  1204. 'allowFeed' => 1,
  1205. 'parent' => 0
  1206. ])
  1207. );
  1208. // write comment
  1209. $installDb->query(
  1210. $installDb->insert('table.comments')->rows([
  1211. 'cid' => 1, 'created' => \Typecho\Date::time(),
  1212. 'author' => 'Typecho',
  1213. 'ownerId' => 1,
  1214. 'url' => 'https://typecho.org',
  1215. 'ip' => '127.0.0.1',
  1216. 'agent' => $options->generator,
  1217. 'text' => '欢迎加入 Typecho 大家族',
  1218. 'type' => 'comment',
  1219. 'status' => 'approved',
  1220. 'parent' => 0
  1221. ])
  1222. );
  1223. // write options
  1224. foreach (install_get_default_options() as $key => $value) {
  1225. // mark installing finished
  1226. if ($key == 'installed') {
  1227. $value = 1;
  1228. }
  1229. $installDb->query(
  1230. $installDb->insert('table.options')->rows(['name' => $key, 'user' => 0, 'value' => $value])
  1231. );
  1232. }
  1233. } catch (\Typecho\Db\Exception $e) {
  1234. install_raise_error($e->getMessage());
  1235. }
  1236. $parts = parse_url($options->loginAction);
  1237. $parts['query'] = http_build_query([
  1238. 'name' => $config['userName'],
  1239. 'password' => $config['userPassword'],
  1240. 'referer' => $options->adminUrl
  1241. ]);
  1242. $loginUrl = \Typecho\Common::buildUrl($parts);
  1243. install_success(0, [
  1244. $config['userName'],
  1245. $config['userPassword'],
  1246. \Widget\Security::alloc()->getTokenUrl($loginUrl, $request->getReferer()),
  1247. $config['userUrl']
  1248. ]);
  1249. }
  1250. /**
  1251. * dispatch install action
  1252. *
  1253. */
  1254. function install_dispatch()
  1255. {
  1256. // disable root url on cli mode
  1257. if (install_is_cli()) {
  1258. define('__TYPECHO_ROOT_URL__', 'http://localhost');
  1259. }
  1260. // init default options
  1261. $options = \Widget\Options::alloc(install_get_default_options());
  1262. \Widget\Init::alloc();
  1263. // display version
  1264. if (install_is_cli()) {
  1265. echo $options->generator . "\n";
  1266. echo 'PHP ' . PHP_VERSION . "\n";
  1267. }
  1268. // install finished yet
  1269. if (
  1270. install_check('config')
  1271. && install_check('db_structure')
  1272. && install_check('db_data')
  1273. ) {
  1274. // redirect to siteUrl if not cli
  1275. if (!install_is_cli()) {
  1276. install_redirect($options->siteUrl);
  1277. }
  1278. exit(1);
  1279. }
  1280. if (install_is_cli()) {
  1281. install_step_1_perform();
  1282. } else {
  1283. $request = \Typecho\Request::getInstance();
  1284. $step = $request->get('step');
  1285. $action = 1;
  1286. switch (true) {
  1287. case $step == 2:
  1288. if (!install_check('db_structure')) {
  1289. $action = 2;
  1290. } else {
  1291. install_redirect('install.php?step=3');
  1292. }
  1293. break;
  1294. case $step == 3:
  1295. if (install_check('db_structure')) {
  1296. $action = 3;
  1297. } else {
  1298. install_redirect('install.php?step=2');
  1299. }
  1300. break;
  1301. default:
  1302. break;
  1303. }
  1304. $method = 'install_step_' . $action;
  1305. if ($request->isPost()) {
  1306. $method .= '_perform';
  1307. $method();
  1308. exit;
  1309. }
  1310. ?>
  1311. <!DOCTYPE HTML>
  1312. <html>
  1313. <head>
  1314. <meta charset="<?php _e('UTF-8'); ?>" />
  1315. <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no">
  1316. <title><?php _e('Typecho 安装程序'); ?></title>
  1317. <link rel="stylesheet" type="text/css" href="<?php $options->adminStaticUrl('css', 'normalize.css') ?>" />
  1318. <link rel="stylesheet" type="text/css" href="<?php $options->adminStaticUrl('css', 'grid.css') ?>" />
  1319. <link rel="stylesheet" type="text/css" href="<?php $options->adminStaticUrl('css', 'style.css') ?>" />
  1320. <link rel="stylesheet" type="text/css" href="<?php $options->adminStaticUrl('css', 'install.css') ?>" />
  1321. <script src="<?php $options->adminStaticUrl('js', 'jquery.js'); ?>"></script>
  1322. </head>
  1323. <body>
  1324. <div class="body container">
  1325. <h1><a href="https://typecho.org" target="_blank" class="i-logo">Typecho</a></h1>
  1326. <?php $method(); ?>
  1327. </div>
  1328. </body>
  1329. </html>
  1330. <?php
  1331. }
  1332. }
  1333. install_dispatch();