blessing-skin-server/app/Http/Controllers/SetupController.php

290 lines
8.5 KiB
PHP
Raw Normal View History

<?php
namespace App\Http\Controllers;
2018-07-21 16:55:36 +08:00
use DB;
use Log;
2017-08-05 15:10:08 +08:00
use File;
use Option;
use Schema;
use Artisan;
use Storage;
use App\Models\User;
use Illuminate\Http\Request;
use Composer\Semver\Comparator;
2016-11-21 21:50:24 +08:00
use App\Exceptions\PrettyPageException;
class SetupController extends Controller
{
2019-04-09 13:21:31 +08:00
public function welcome()
{
// @codeCoverageIgnoreStart
if (! File::exists(base_path('.env'))) {
File::copy(base_path('.env.example'), base_path('.env'));
}
// @codeCoverageIgnoreEnd
return view('setup.wizard.welcome');
}
2018-07-21 16:55:36 +08:00
public function database(Request $request)
{
if ($request->isMethod('get')) {
try {
DB::getPdo();
2019-04-19 19:36:36 +08:00
return redirect('setup/info');
2019-03-30 16:22:08 +08:00
// @codeCoverageIgnoreStart
} catch (\Exception $e) {
return view('setup.wizard.database');
2019-03-30 16:22:08 +08:00
// @codeCoverageIgnoreEnd
}
}
2018-07-21 16:55:36 +08:00
config([
'database.connections.temp.driver' => $request->input('type'),
'database.connections.temp.host' => $request->input('host'),
'database.connections.temp.port' => $request->input('port'),
'database.connections.temp.username' => $request->input('username'),
'database.connections.temp.password' => $request->input('password'),
2019-04-09 13:21:31 +08:00
'database.connections.temp.database' => $request->input('db'),
'database.connections.temp.prefix' => $request->input('prefix'),
2018-07-21 16:55:36 +08:00
]);
try {
DB::connection('temp')->getPdo();
} catch (\Exception $e) {
$msg = iconv('gbk', 'utf-8', $e->getMessage());
$type = humanize_db_type($request->input('type'));
throw new PrettyPageException(
trans('setup.database.connection-error', compact('msg', 'type')),
$e->getCode()
);
2018-02-22 21:26:23 +08:00
}
2019-03-31 10:33:03 +08:00
$content = File::get(base_path('.env'));
2019-04-09 13:21:31 +08:00
$content = preg_replace(
'/DB_CONNECTION.+/',
2018-07-21 16:55:36 +08:00
'DB_CONNECTION = '.$request->input('type'),
$content
);
2019-04-09 13:21:31 +08:00
$content = preg_replace(
'/DB_HOST.+/',
2018-07-21 16:55:36 +08:00
'DB_HOST = '.$request->input('host'),
$content
);
2019-04-09 13:21:31 +08:00
$content = preg_replace(
'/DB_PORT.+/',
2018-07-21 16:55:36 +08:00
'DB_PORT = '.$request->input('port'),
$content
);
2019-04-09 13:21:31 +08:00
$content = preg_replace(
'/DB_DATABASE.+/',
2018-07-21 16:55:36 +08:00
'DB_DATABASE = '.$request->input('db'),
$content
);
2019-04-09 13:21:31 +08:00
$content = preg_replace(
'/DB_USERNAME.+/',
2018-07-21 16:55:36 +08:00
'DB_USERNAME = '.$request->input('username'),
$content
);
2019-04-09 13:21:31 +08:00
$content = preg_replace(
'/DB_PASSWORD.+/',
2018-07-21 16:55:36 +08:00
'DB_PASSWORD = '.$request->input('password'),
$content
);
2019-04-09 13:21:31 +08:00
$content = preg_replace(
'/DB_PREFIX.+/',
2018-07-21 16:55:36 +08:00
'DB_PREFIX = '.$request->input('prefix'),
$content
);
2019-03-31 10:33:03 +08:00
File::put(base_path('.env'), $content);
2018-07-21 16:55:36 +08:00
return redirect('setup/info');
}
public function info()
{
$existingTables = static::checkTablesExist([], true);
// Not installed completely
if (count($existingTables) > 0) {
Log::info('Remaining tables detected, exit setup wizard now', [$existingTables]);
$existingTables = array_map(function ($item) {
return get_db_config()['prefix'].$item;
}, $existingTables);
throw new PrettyPageException(trans('setup.database.table-already-exists', ['tables' => json_encode($existingTables)]), 1);
}
2018-03-11 12:36:23 +08:00
// @codeCoverageIgnoreStart
2018-02-22 23:07:23 +08:00
if (! function_exists('escapeshellarg')) {
throw new PrettyPageException(trans('setup.disabled-functions.escapeshellarg'), 1);
}
2018-03-11 12:36:23 +08:00
// @codeCoverageIgnoreEnd
2018-02-22 23:07:23 +08:00
return view('setup.wizard.info');
}
public function finish(Request $request)
{
$data = $this->validate($request, [
'email' => 'required|email',
2018-07-20 17:23:54 +08:00
'nickname' => 'required|no_special_chars|max:255',
2018-06-18 21:50:32 +08:00
'password' => 'required|min:8|max:32|confirmed',
'site_name' => 'required',
]);
2017-11-20 19:56:24 +08:00
if ($request->has('generate_random')) {
2019-03-19 10:13:57 +08:00
Artisan::call('key:random');
Artisan::call('salt:random');
}
2019-04-23 10:05:58 +08:00
Artisan::call('jwt:secret', ['--no-interaction' => true]);
2019-04-25 23:24:24 +08:00
Artisan::call('passport:keys', ['--no-interaction' => true]);
// Create tables
Artisan::call('migrate', [
'--force' => true,
'--path' => [
'database/migrations',
'vendor/laravel/passport/database/migrations'
]
]);
Log::info('[SetupWizard] Tables migrated.');
Option::set('site_name', $request->input('site_name'));
$siteUrl = url('/');
if (ends_with($siteUrl, '/index.php')) {
2017-11-20 19:56:24 +08:00
$siteUrl = substr($siteUrl, 0, -10); // @codeCoverageIgnore
}
Option::set('site_url', $siteUrl);
2018-02-16 17:31:04 +08:00
// Register super admin
$user = new User;
$user->email = $data['email'];
2018-07-20 17:23:54 +08:00
$user->nickname = $data['nickname'];
$user->score = option('user_initial_score');
$user->avatar = 0;
2019-07-30 15:12:31 +08:00
$user->password = app('cipher')->hash($data['password'], config('secure.salt'));
2018-08-17 22:54:26 +08:00
$user->ip = get_client_ip();
$user->permission = User::SUPER_ADMIN;
2018-08-17 22:54:26 +08:00
$user->register_at = get_datetime_string();
$user->last_sign_at = get_datetime_string(time() - 86400);
2018-08-21 11:03:57 +08:00
$user->verified = true;
$user->save();
$this->createDirectories();
return view('setup.wizard.finish')->with([
'email' => $request->input('email'),
'password' => $request->input('password'),
]);
}
public function update()
{
if (Comparator::lessThanOrEqualTo(config('app.version'), option('version'))) {
2018-02-16 17:31:04 +08:00
// No updates available
return view('setup.locked');
}
return view('setup.updates.welcome');
}
public function doUpdate()
{
$resource = opendir(database_path('update_scripts'));
$updateScriptExist = false;
while ($filename = @readdir($resource)) {
if ($filename != '.' && $filename != '..') {
preg_match('/update-(.*)-to-(.*).php/', $filename, $matches);
2018-02-16 17:31:04 +08:00
// Skip if the file is not valid or expired
if (! isset($matches[2]) ||
Comparator::lessThan($matches[2], config('app.version'))) {
continue;
}
2019-05-03 08:46:15 +08:00
$tips = require database_path('update_scripts')."/$filename";
$updateScriptExist = true;
}
}
closedir($resource);
foreach (config('options') as $key => $value) {
2018-02-16 17:31:04 +08:00
if (! Option::has($key)) {
Option::set($key, $value);
2018-02-16 17:31:04 +08:00
}
}
2019-04-22 22:56:24 +08:00
Option::set('version', config('app.version'));
2019-05-03 08:46:15 +08:00
Artisan::call('view:clear');
2017-01-13 22:34:06 +08:00
2019-05-03 23:44:04 +08:00
return view('setup.updates.success', ['tips' => $tips ?? []]);
}
/**
* Check if the given tables exist in current database.
*
* @param array $tables
* @param bool $returnExisting
* @return bool|array
*/
public static function checkTablesExist($tables = [], $returnExistingTables = false)
{
$existingTables = [];
$tables = $tables ?: [
'users',
'user_closet',
'players',
'textures',
'options',
'reports',
];
2018-02-16 17:31:04 +08:00
foreach ($tables as $tableName) {
if (Schema::hasTable($tableName)) {
$existingTables[] = $tableName;
}
}
if (count($existingTables) == count($tables)) {
return $returnExistingTables ? $existingTables : true;
} else {
return $returnExistingTables ? $existingTables : false;
}
}
public static function checkDirectories()
{
$directories = ['storage/textures', 'plugins'];
try {
foreach ($directories as $dir) {
2018-02-16 17:31:04 +08:00
if (! Storage::disk('root')->has($dir)) {
// Try to mkdir
if (! Storage::disk('root')->makeDirectory($dir)) {
return false;
2018-02-16 17:31:04 +08:00
}
}
}
return true;
} catch (\Exception $e) {
return false;
}
}
protected function createDirectories()
{
return self::checkDirectories();
}
}