| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2018-05-11 10:08:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Installer.php | 
					
						
							|  |  |  |  * Copyright (c) 2018 thegrumpydictator@gmail.com | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is part of Firefly III. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Firefly III is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Firefly III is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with Firefly III. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-19 13:23:26 +01:00
										 |  |  | declare(strict_types=1); | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Http\Middleware; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Closure; | 
					
						
							|  |  |  | use DB; | 
					
						
							|  |  |  | use FireflyIII\Exceptions\FireflyException; | 
					
						
							|  |  |  | use Illuminate\Database\QueryException; | 
					
						
							|  |  |  | use Log; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class Installer | 
					
						
							| 
									
										
										
										
											2019-02-13 17:38:41 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-09-07 20:12:22 +02:00
										 |  |  |  * @codeCoverageIgnore | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | class Installer | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Handle an incoming request. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @throws FireflyException | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param  \Illuminate\Http\Request $request | 
					
						
							|  |  |  |      * @param  \Closure                 $next | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return mixed | 
					
						
							| 
									
										
										
										
											2018-07-17 22:21:03 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @SuppressWarnings(PHPMD.ExcessiveMethodLength) | 
					
						
							|  |  |  |      * @SuppressWarnings(PHPMD.CyclomaticComplexity) | 
					
						
							| 
									
										
										
										
											2018-09-07 20:12:22 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function handle($request, Closure $next) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |         // ignore installer in test environment.
 | 
					
						
							| 
									
										
										
										
											2018-12-15 07:59:02 +01:00
										 |  |  |         if ('testing' === config('app.env')) { | 
					
						
							| 
									
										
										
										
											2018-03-07 21:04:10 +01:00
										 |  |  |             return $next($request); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |         // don't run installer when already in installer.
 | 
					
						
							| 
									
										
										
										
											2018-03-10 22:38:20 +01:00
										 |  |  |         $url    = $request->url(); | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |         $strpos = stripos($url, '/install'); | 
					
						
							| 
									
										
										
										
											2018-07-08 07:59:58 +02:00
										 |  |  |         if (!(false === $strpos)) { | 
					
						
							| 
									
										
										
										
											2018-03-07 20:56:52 +01:00
										 |  |  |             Log::debug(sprintf('URL is %s, will NOT run installer middleware', $url)); | 
					
						
							| 
									
										
										
										
											2018-03-10 22:38:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |             return $next($request); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // run installer when no tables are present,
 | 
					
						
							|  |  |  |         // or when old scheme version
 | 
					
						
							|  |  |  |         // or when old firefly version
 | 
					
						
							|  |  |  |         if ($this->hasNoTables() || $this->oldDBVersion() || $this->oldVersion()) { | 
					
						
							|  |  |  |             return response()->redirectTo(route('installer.index')); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // update scheme version
 | 
					
						
							|  |  |  |         // update firefly version
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $next($request); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Is access denied error. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param string $message | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function isAccessDenied(string $message): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return !(false === stripos($message, 'Access denied')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Is no tables exist error. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param string $message | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function noTablesExist(string $message): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return !(false === stripos($message, 'Base table or view not found')); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if the tables are created and accounted for. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      * @throws FireflyException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private function hasNoTables(): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         Log::debug('Now in routine hasNoTables()'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |         try { | 
					
						
							|  |  |  |             DB::table('users')->count(); | 
					
						
							|  |  |  |         } catch (QueryException $e) { | 
					
						
							|  |  |  |             $message = $e->getMessage(); | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |             Log::error(sprintf('Error message trying to access users-table: %s', $message)); | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |             if ($this->isAccessDenied($message)) { | 
					
						
							|  |  |  |                 throw new FireflyException('It seems your database configuration is not correct. Please verify the username and password in your .env file.'); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if ($this->noTablesExist($message)) { | 
					
						
							|  |  |  |                 // redirect to UpdateController
 | 
					
						
							|  |  |  |                 Log::warning('There are no Firefly III tables present. Redirect to migrate routine.'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |                 return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |             throw new FireflyException(sprintf('Could not access the database: %s', $message)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |         Log::debug('Everything seems OK with the tables.'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if the "db_version" variable is correct. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private function oldDBVersion(): bool | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |         // older version in config than database?
 | 
					
						
							| 
									
										
										
										
											2018-04-02 15:10:40 +02:00
										 |  |  |         $configVersion = (int)config('firefly.db_version'); | 
					
						
							| 
									
										
										
										
											2019-02-13 17:38:41 +01:00
										 |  |  |         $dbVersion     = (int)app('fireflyconfig')->getFresh('db_version', 1)->data; | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |         if ($configVersion > $dbVersion) { | 
					
						
							| 
									
										
										
										
											2018-03-10 22:38:20 +01:00
										 |  |  |             Log::warning( | 
					
						
							|  |  |  |                 sprintf( | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |                     'The current configured version (%d) is older than the required version (%d). Redirect to migrate routine.', $dbVersion, $configVersion | 
					
						
							| 
									
										
										
										
											2018-03-10 22:38:20 +01:00
										 |  |  |                 ) | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |         Log::info(sprintf('Configured DB version (%d) equals expected DB version (%d)', $dbVersion, $configVersion)); | 
					
						
							| 
									
										
										
										
											2018-03-10 22:38:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |      * Check if the "firefly_version" variable is correct. | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |     private function oldVersion(): bool | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |         // version compare thing.
 | 
					
						
							|  |  |  |         $configVersion = (string)config('firefly.version'); | 
					
						
							|  |  |  |         $dbVersion     = (string)app('fireflyconfig')->getFresh('ff3_version', '1.0')->data; | 
					
						
							|  |  |  |         if (1 === version_compare($configVersion, $dbVersion)) { | 
					
						
							|  |  |  |             Log::warning( | 
					
						
							|  |  |  |                 sprintf( | 
					
						
							|  |  |  |                     'The current configured Firefly III version (%s) is older than the required version (%s). Redirect to migrate routine.', $dbVersion, $configVersion | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-17 09:06:45 +01:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         Log::info(sprintf('Installed Firefly III version (%s) equals expected Firefly III version (%s)', $dbVersion, $configVersion)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2018-03-07 20:21:36 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } |