Migrate all configs into the env.php file #1075 (#1128)

* Migrate all configs into the env.php file #1075

* Style fixes

* Fix config names in CreateConfig

* Fix installer/installer error messages
This commit is contained in:
Nabeel S 2021-04-10 17:19:45 -04:00 committed by GitHub
parent 6c3992e781
commit ac64a5db75
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
17 changed files with 247 additions and 81 deletions

View File

@ -8,7 +8,7 @@ APP_LOCALE="en"
PHPVMS_INSTALLED="true" PHPVMS_INSTALLED="true"
APP_LOG="daily" APP_LOG="daily"
APP_LOG_LEVEL="debug" LOG_LEVEL="debug"
APP_LOG_MAX_FILES="3" APP_LOG_MAX_FILES="3"
DB_CONNECTION="mysql" DB_CONNECTION="mysql"

1
.gitignore vendored
View File

@ -74,4 +74,5 @@ error_log
.sass-cache .sass-cache
.DS_Store .DS_Store
/config.php /config.php
/config.bak.php
/VERSION /VERSION

View File

@ -28,6 +28,8 @@ RedirectMatch 403 ^/composer.phar
RedirectMatch 403 ^/env.php.*?$ RedirectMatch 403 ^/env.php.*?$
RedirectMatch 403 ^/env.php RedirectMatch 403 ^/env.php
RedirectMatch 403 ^/env.php$ RedirectMatch 403 ^/env.php$
RedirectMatch 403 ^/config.php$
RedirectMatch 403 ^/config.bak.php$
RedirectMatch 403 ^/Makefile RedirectMatch 403 ^/Makefile
RedirectMatch 403 ^/package.json RedirectMatch 403 ^/package.json
RedirectMatch 403 ^/package-lock.json RedirectMatch 403 ^/package-lock.json

View File

@ -2,7 +2,7 @@ FROM php:7.4-fpm-alpine
WORKDIR /var/www/ WORKDIR /var/www/
RUN apk add gmp-dev RUN apk add gmp-dev icu-dev
RUN curl --silent --show-error https://getcomposer.org/installer | php RUN curl --silent --show-error https://getcomposer.org/installer | php
# Copy any config files in # Copy any config files in
@ -12,6 +12,7 @@ RUN ln -sf /dev/stderr /var/log/fpm-error.log
RUN docker-php-ext-install \ RUN docker-php-ext-install \
calendar \ calendar \
intl \
pdo_mysql \ pdo_mysql \
gmp \ gmp \
opcache && \ opcache && \

View File

@ -4,9 +4,9 @@ namespace App\Console\Commands;
use App; use App;
use App\Contracts\Command; use App\Contracts\Command;
use App\Services\Installer\ConfigService;
use App\Services\Installer\SeederService; use App\Services\Installer\SeederService;
use DatabaseSeeder; use DatabaseSeeder;
use Modules\Installer\Services\ConfigService;
/** /**
* Create the config files * Create the config files
@ -81,13 +81,13 @@ class CreateConfigs extends Command
$this->info('Regenerating the config files'); $this->info('Regenerating the config files');
$cfgSvc->createConfigFiles([ $cfgSvc->createConfigFiles([
'APP_ENV' => 'dev', 'APP_ENV' => 'dev',
'SITE_NAME' => $this->argument('name'), 'SITE_NAME' => $this->argument('name'),
'DB_CONN' => 'mysql', 'DB_CONNECTION' => 'mysql',
'DB_HOST' => $this->argument('db_host'), 'DB_HOST' => $this->argument('db_host'),
'DB_NAME' => $this->argument('db_name'), 'DB_DATABASE' => $this->argument('db_name'),
'DB_USER' => $this->argument('db_user'), 'DB_USERNAME' => $this->argument('db_user'),
'DB_PASS' => $this->argument('db_pass'), 'DB_PASSWORD' => $this->argument('db_pass'),
]); ]);
$this->info('Config files generated!'); $this->info('Config files generated!');

View File

@ -79,9 +79,9 @@ class DevInstall extends Command
$this->info('Regenerating the config files'); $this->info('Regenerating the config files');
$cfgSvc->createConfigFiles([ $cfgSvc->createConfigFiles([
'APP_ENV' => 'dev', 'APP_ENV' => 'dev',
'SITE_NAME' => 'phpvms test', 'SITE_NAME' => 'phpvms test',
'DB_CONN' => 'sqlite', 'DB_CONNECTION' => 'sqlite',
]); ]);
$this->info('Config files generated!'); $this->info('Config files generated!');

View File

@ -0,0 +1,25 @@
<?php
namespace App\Console\Commands;
use App\Contracts\Command;
use App\Services\Installer\ConfigService;
/**
* Command to rewrite the config files
*/
class RewriteConfigs extends Command
{
protected $signature = 'phpvms:rewrite-configs';
protected $description = 'Rewrite the config files';
/**
* Run dev related commands
*/
public function handle()
{
/** @var ConfigService $configSvc */
$configSvc = app(ConfigService::class);
$configSvc->rewriteConfigFiles();
}
}

View File

@ -0,0 +1,21 @@
<?php
use App\Contracts\Migration;
use App\Services\Installer\ConfigService;
/**
* Migrate the configuration files
*/
class MigrateConfigs extends Migration
{
public function up()
{
/** @var ConfigService $configSvc */
$configSvc = app(ConfigService::class);
$configSvc->rewriteConfigFiles();
}
public function down()
{
}
}

View File

@ -203,21 +203,21 @@ class InstallerController extends Controller
Log::error('Testing db before writing configs failed'); Log::error('Testing db before writing configs failed');
Log::error($e->getMessage()); Log::error($e->getMessage());
Flash::error($e->getMessage()); flash()->error($e->getMessage());
return redirect(route('installer.step2'))->withInput(); return redirect(route('installer.step2'))->withInput();
} }
// Now write out the env file // Now write out the env file
$attrs = [ $attrs = [
'SITE_NAME' => $request->post('site_name'), 'SITE_NAME' => $request->post('site_name'),
'SITE_URL' => $request->post('site_url'), 'SITE_URL' => $request->post('site_url'),
'DB_CONN' => $request->post('db_conn'), 'DB_CONNECTION' => $request->post('db_conn'),
'DB_HOST' => $request->post('db_host'), 'DB_HOST' => $request->post('db_host'),
'DB_PORT' => $request->post('db_port'), 'DB_PORT' => $request->post('db_port'),
'DB_NAME' => $request->post('db_name'), 'DB_DATABASE' => $request->post('db_name'),
'DB_USER' => $request->post('db_user'), 'DB_USERNAME' => $request->post('db_user'),
'DB_PASS' => $request->post('db_pass'), 'DB_PASSWORD' => $request->post('db_pass'),
'DB_PREFIX' => $request->post('db_prefix'), 'DB_PREFIX' => $request->post('db_prefix'),
]; ];
/* /*
@ -231,7 +231,7 @@ class InstallerController extends Controller
Log::error('Config files failed to write'); Log::error('Config files failed to write');
Log::error($e->getMessage()); Log::error($e->getMessage());
Flash::error($e->getMessage()); flash()->error($e->getMessage());
return redirect(route('installer.step2'))->withInput(); return redirect(route('installer.step2'))->withInput();
} }
@ -257,7 +257,7 @@ class InstallerController extends Controller
Log::error('Error on db setup: '.$e->getMessage()); Log::error('Error on db setup: '.$e->getMessage());
//dd($e); //dd($e);
$this->envSvc->removeConfigFiles(); $this->envSvc->removeConfigFiles();
Flash::error($e->getMessage()); flash()->error($e->getMessage());
return redirect(route('installer.step2'))->withInput(); return redirect(route('installer.step2'))->withInput();
} }

View File

@ -15,34 +15,49 @@ use Symfony\Component\HttpFoundation\File\Exception\FileException;
class ConfigService extends Service class ConfigService extends Service
{ {
protected static $defaultValues = [
'APP_ENV' => 'dev',
'APP_KEY' => '',
'APP_DEBUG' => true,
'APP_LOCALE' => 'en',
'DEBUG_TOOLBAR' => false,
'SITE_NAME' => '',
'SITE_URL' => 'http://phpvms.test',
'DB_CONNECTION' => '',
'DB_HOST' => '',
'DB_PORT' => 3306,
'DB_DATABASE' => '',
'DB_USERNAME' => '',
'DB_PASSWORD' => '',
'DB_PREFIX' => '',
'DB_EMULATE_PREPARES' => false,
'CACHE_DRIVER' => 'array',
'CACHE_PREFIX' => '',
'MAIL_DRIVER' => 'smtp',
'MAIL_HOST' => '',
'MAIL_PORT' => 587,
'MAIL_ENCRYPTION' => '',
'MAIL_USERNAME' => '',
'MAIL_PASSWORD' => '',
'MAIL_FROM_NAME' => 'phpVMS Admin',
'MAIL_FROM_ADDRESS' => 'no-reply@phpvms.net',
];
/** /**
* Create the .env file * Create the .env file. This is called by an initial install
* *
* @param $attrs * @param $attrs
* *
* @throws \Symfony\Component\HttpFoundation\File\Exception\FileException * @throws FileException
* *
* @return bool * @return bool
*/ */
public function createConfigFiles($attrs): bool public function createConfigFiles($attrs): bool
{ {
$opts = [ $opts = array_merge(static::$defaultValues, $attrs);
'APP_ENV' => 'dev', if (empty($opts['APP_KEY'])) {
'APP_KEY' => $this->createAppKey(), $opts['APP_KEY'] = $this->createAppKey();
'SITE_NAME' => '', }
'SITE_URL' => 'http://phpvms.test',
'CACHE_PREFIX' => '',
'DB_CONN' => '',
'DB_HOST' => '',
'DB_PORT' => 3306,
'DB_NAME' => '',
'DB_USER' => '',
'DB_PASS' => '',
'DB_PREFIX' => '',
'DB_EMULATE_PREPARES' => false,
];
$opts = array_merge($opts, $attrs);
$opts = $this->determinePdoOptions($opts); $opts = $this->determinePdoOptions($opts);
$opts = $this->configCacheDriver($opts); $opts = $this->configCacheDriver($opts);
@ -53,6 +68,61 @@ class ConfigService extends Service
return true; return true;
} }
/**
* Rewrite the config files - this means mapping the values that are currently
* loaded in the config and rewriting them into the env.php file, and then renaming
* the config.php files to config.bak.php
*
* This is called from the migrations which removes the old config.php file
*/
public function rewriteConfigFiles()
{
/*$cfg_file = App::environmentPath().'/config.php';
if (!file_exists($cfg_file)) {
Log::info('Main config.php file is missing, migration already completed');
return;
}*/
$db_opts = config('database.connections.mysql.options');
$emulate_prepares = $db_opts[PDO::ATTR_EMULATE_PREPARES] ? 'true' : 'false';
$opts = array_merge(static::$defaultValues, [
'APP_ENV' => config('app.env'),
'APP_KEY' => config('app.key'),
'APP_DEBUG' => config('app.debug') ? 'true' : 'false',
'APP_LOCALE' => config('app.locale'),
'DEBUG_TOOLBAR' => config('app.debug_toolbar') ? 'true' : 'false',
'SITE_NAME' => config('app.name'),
'SITE_URL' => config('app.url'),
'DB_CONNECTION' => config('database.default'),
'DB_HOST' => config('database.connections.mysql.host'),
'DB_PORT' => config('database.connections.mysql.port'),
'DB_DATABASE' => config('database.connections.mysql.database'),
'DB_USERNAME' => config('database.connections.mysql.username'),
'DB_PASSWORD' => config('database.connections.mysql.password'),
'DB_PREFIX' => config('database.connections.mysql.prefix'),
'DB_EMULATE_PREPARES' => $emulate_prepares,
'CACHE_DRIVER' => config('cache.default'),
'CACHE_PREFIX' => config('cache.prefix'),
'MAIL_DRIVER' => config('mail.default'),
'MAIL_HOST' => config('mail.mailers.smtp.host'),
'MAIL_PORT' => config('mail.mailers.smtp.port'),
'MAIL_ENCRYPTION' => config('mail.mailers.smtp.encryption'),
'MAIL_USERNAME' => config('mail.mailers.smtp.username'),
'MAIL_PASSWORD' => config('mail.mailers.smtp.password'),
'MAIL_FROM_NAME' => config('mail.from.name'),
'MAIL_FROM_ADDRESS' => config('mail.from.address'),
]);
$this->writeConfigFiles($opts);
// Rename the old config file
$cfg_file = App::environmentPath().'/config.php';
if (file_exists($cfg_file)) {
rename($cfg_file, App::environmentPath().'/config.bak.php');
}
}
/** /**
* Update the environment file and update certain keys/values * Update the environment file and update certain keys/values
* *
@ -97,7 +167,7 @@ class ConfigService extends Service
*/ */
protected function createAppKey(): string protected function createAppKey(): string
{ {
return base64_encode(Encrypter::generateKey(config('app.cipher'))); return 'base64:'.base64_encode(Encrypter::generateKey(config('app.cipher')));
} }
/** /**
@ -110,14 +180,14 @@ class ConfigService extends Service
*/ */
protected function determinePdoOptions($opts) protected function determinePdoOptions($opts)
{ {
if ($opts['DB_CONN'] !== 'mysql') { if ($opts['DB_CONNECTION'] !== 'mysql') {
return $opts; return $opts;
} }
$dsn = "mysql:host=$opts[DB_HOST];port=$opts[DB_PORT];"; $dsn = "mysql:host=$opts[DB_HOST];port=$opts[DB_PORT];";
Log::info('Connection string: '.$dsn); Log::info('Connection string: '.$dsn);
$conn = new PDO($dsn, $opts['DB_USER'], $opts['DB_PASS']); $conn = new PDO($dsn, $opts['DB_USERNAME'], $opts['DB_PASSWORD']);
$version = strtolower($conn->getAttribute(PDO::ATTR_SERVER_VERSION)); $version = strtolower($conn->getAttribute(PDO::ATTR_SERVER_VERSION));
Log::info('Detected DB Version: '.$version); Log::info('Detected DB Version: '.$version);
@ -171,7 +241,7 @@ class ConfigService extends Service
{ {
// If we're setting up a database, then also setup // If we're setting up a database, then also setup
// the default queue driver to use the database // the default queue driver to use the database
if ($opts['DB_CONN'] === 'mysql' || $opts['DB_CONN'] === 'postgres') { if ($opts['DB_CONNECTION'] === 'mysql' || $opts['DB_CONNECTION'] === 'postgres') {
$opts['QUEUE_DRIVER'] = 'database'; $opts['QUEUE_DRIVER'] = 'database';
} else { } else {
$opts['QUEUE_DRIVER'] = 'sync'; $opts['QUEUE_DRIVER'] = 'sync';
@ -210,14 +280,13 @@ class ConfigService extends Service
* *
* @param $opts * @param $opts
* *
* @throws \Symfony\Component\HttpFoundation\File\Exception\FileException * @throws FileException
*/ */
protected function writeConfigFiles($opts) protected function writeConfigFiles($opts)
{ {
Stub::setBasePath(resource_path('/stubs/installer')); Stub::setBasePath(resource_path('/stubs/installer'));
$env_file = App::environmentFilePath(); $env_file = App::environmentFilePath();
if (file_exists($env_file) && !is_writable($env_file)) { if (file_exists($env_file) && !is_writable($env_file)) {
Log::error('Permissions on existing env.php is not writable'); Log::error('Permissions on existing env.php is not writable');
@ -239,7 +308,7 @@ class ConfigService extends Service
* Next write out the config file. If there's an error here, * Next write out the config file. If there's an error here,
* then throw an exception but delete the env file first * then throw an exception but delete the env file first
*/ */
try { /*try {
$stub = new Stub('/config.stub', $opts); $stub = new Stub('/config.stub', $opts);
$stub->render(); $stub->render();
$stub->saveTo(App::environmentPath(), 'config.php'); $stub->saveTo(App::environmentPath(), 'config.php');
@ -247,6 +316,6 @@ class ConfigService extends Service
unlink(App::environmentPath().'/'.App::environmentFile()); unlink(App::environmentPath().'/'.App::environmentFile());
throw new FileException('Couldn\'t write config.php. ('.$e.')'); throw new FileException('Couldn\'t write config.php. ('.$e.')');
} }*/
} }
} }

View File

@ -15,7 +15,7 @@ return [
'debug' => env('APP_DEBUG', true), 'debug' => env('APP_DEBUG', true),
'url' => env('APP_URL', ''), 'url' => env('APP_URL', ''),
'version' => '7.0.0', 'version' => '7.0.0',
'debug_toolbar' => false, 'debug_toolbar' => env('DEBUG_TOOLBAR', false),
'locale' => env('APP_LOCALE', 'en'), 'locale' => env('APP_LOCALE', 'en'),
'fallback_locale' => 'en', 'fallback_locale' => 'en',

View File

@ -1,5 +1,7 @@
<?php <?php
use App\Console\Logger;
return [ return [
/* /*
|-------------------------------------------------------------------------- |--------------------------------------------------------------------------
@ -44,18 +46,18 @@ return [
'single' => [ 'single' => [
'driver' => 'single', 'driver' => 'single',
'path' => storage_path('logs/laravel.log'), 'path' => storage_path('logs/laravel.log'),
'level' => 'debug', 'level' => env('LOG_LEVEL', 'debug'),
], ],
'daily' => [ 'daily' => [
'driver' => 'daily', 'driver' => 'daily',
'path' => storage_path('logs/laravel.log'), 'path' => storage_path('logs/laravel.log'),
'level' => 'debug', 'level' => env('LOG_LEVEL', 'debug'),
'days' => 3, 'days' => 3,
], ],
'cron_rotating' => [ 'cron_rotating' => [
'driver' => 'daily', 'driver' => 'daily',
'path' => storage_path('logs/cron.log'), 'path' => storage_path('logs/cron.log'),
'level' => 'debug', 'level' => env('LOG_LEVEL', 'debug'),
'days' => 3, 'days' => 3,
], ],
'slack' => [ 'slack' => [
@ -67,15 +69,15 @@ return [
], ],
'stdout' => [ 'stdout' => [
'driver' => 'custom', 'driver' => 'custom',
'via' => \App\Console\Logger::class, 'via' => Logger::class,
], ],
'syslog' => [ 'syslog' => [
'driver' => 'syslog', 'driver' => 'syslog',
'level' => 'debug', 'level' => env('LOG_LEVEL', 'debug'),
], ],
'errorlog' => [ 'errorlog' => [
'driver' => 'errorlog', 'driver' => 'errorlog',
'level' => 'debug', 'level' => env('LOG_LEVEL', 'debug'),
], ],
], ],
]; ];

View File

@ -1,5 +1,7 @@
<?php <?php
use App\Models\User;
return [ return [
/* /*
|-------------------------------------------------------------------------- |--------------------------------------------------------------------------
@ -29,7 +31,7 @@ return [
], ],
'stripe' => [ 'stripe' => [
'model' => App\User::class, 'model' => User::class,
'key' => env('STRIPE_KEY'), 'key' => env('STRIPE_KEY'),
'secret' => env('STRIPE_SECRET'), 'secret' => env('STRIPE_SECRET'),
], ],

View File

@ -10,6 +10,7 @@ RUN ln -sf /dev/stderr /var/log/fpm-error.log
RUN docker-php-ext-install \ RUN docker-php-ext-install \
bcmath \ bcmath \
calendar \ calendar \
pdo_mysql \
gmp \ gmp \
opcache intl \
opcache \
pdo_mysql

View File

@ -60,9 +60,9 @@ return [
'mysql' => [ 'mysql' => [
'host' => env('DB_HOST', '$DB_HOST$'), 'host' => env('DB_HOST', '$DB_HOST$'),
'port' => $DB_PORT$, 'port' => $DB_PORT$,
'database' => '$DB_NAME$', 'database' => '$DB_DATABASE$',
'username' => '$DB_USER$', 'username' => '$DB_USERNAME$',
'password' => '$DB_PASS$', 'password' => '$DB_PASSWORD$',
'prefix' => '$DB_PREFIX$', 'prefix' => '$DB_PREFIX$',
'prefix_indexes' => true, 'prefix_indexes' => true,
], ],

View File

@ -1,8 +1,49 @@
#
# THIS FILE MUST BE KEPT SECRET! IT IS BLOCKED IN THE HTACCESS FILE # THIS FILE MUST BE KEPT SECRET! IT IS BLOCKED IN THE HTACCESS FILE
# HOWEVER, THIS DIRECTORY SHOULDN'T BE EXPOSED TO THE PUBLIC AT ALL # HOWEVER, THIS DIRECTORY SHOULDN'T BE EXPOSED TO THE PUBLIC AT ALL
# SEE THE DOCS FOR PROPER (SECURE) INSTALLATION: # SEE THE DOCS FOR PROPER (SECURE) INSTALLATION:
# https://docs.phpvms.net/installation/uploading # https://docs.phpvms.net/installation/uploading
# #
# This is the place to edit your configuration. To change a config that's
# not present here, you need to either edit the file/config in the config
# folder, or change it to read the value from the environment. Something like:
#
# 'some_key' = env('ENVIRONMENT_VARIABLE_KEY_ADDED_BELOW', 'default value')
#
# Various other settings in the configs also read from some environment variables
# by default. You can override those here
APP_KEY=base64:$APP_KEY$ APP_KEY='$APP_KEY$'
SITE_NAME='$SITE_NAME$'
SITE_URL='$SITE_URL$'
APP_ENV=$APP_ENV$
APP_LOCALE=$APP_LOCALE$
LOG_LEVEL=debug
APP_DEBUG=$APP_DEBUG$
DEBUG_TOOLBAR=$DEBUG_TOOLBAR$
# DATABASE SETTINGS
DB_CONNECTION=$DB_CONNECTION$
DB_HOST='$DB_HOST$'
DB_PORT=$DB_PORT$
DB_DATABASE='$DB_DATABASE$'
DB_USERNAME='$DB_USERNAME$'
DB_PASSWORD='$DB_PASSWORD$'
DB_PREFIX='$DB_PREFIX$'
DB_EMULATE_PREPARES=$DB_EMULATE_PREPARES$
DB_SOCKET=
# CACHE SETTINGS
CACHE_DRIVER=$CACHE_DRIVER$
CACHE_PREFIX=$CACHE_PREFIX$
# EMAIL SETTINGS
# Look at the available mail configs in config/mail.php
# Also refer to the Laravel docs here: https://laravel.com/docs/8.x/mail
MAIL_DRIVER=$MAIL_DRIVER$
MAIL_FROM_NAME='$MAIL_FROM_NAME$'
MAIL_FROM_ADDRESS='$MAIL_FROM_ADDRESS$'
MAIL_HOST=$MAIL_HOST$
MAIL_PORT=$MAIL_PORT$
MAIL_ENCRYPTION=$MAIL_ENCRYPTION$
MAIL_USERNAME=$MAIL_USERNAME$
MAIL_PASSWORD=$MAIL_PASSWORD$

View File

@ -1,19 +1,20 @@
@if (session()->has('flash_notification.message')) @foreach (collect(session('flash_notification', collect()))->toArray() as $message)
@if (session()->has('flash_notification.overlay')) @if (is_string($message))
@include('flash::modal', [ <div class="alert alert-error">{!! $message !!}</div>
'modalClass' => 'flash-modal',
'title' => session('flash_notification.title'),
'body' => session('flash_notification.message')
])
@else @else
<div class="alert <div class="alert alert-{{ $message['level'] }}
alert-{{ session('flash_notification.level') }} {{ $message['important'] ? 'alert-important' : '' }}"
{{ session()->has('flash_notification.important') ? 'alert-important' : '' }}"> role="alert">
@if(session()->has('flash_notification.important')) @if ($message['important'])
<button type="button" class="close" data-dismiss="alert">&times;</button> <button type="button"
class="close"
data-dismiss="alert"
aria-hidden="true">&times;</button>
@endif @endif
{{ session('flash_notification.message') }} {!! $message['message'] !!}
</div> </div>
@endif @endif
@endif @endforeach
{{ session()->forget('flash_notification') }}