2020-02-05 05:06:03 +01:00
|
|
|
<?php
|
|
|
|
/**
|
2020-09-06 11:38:10 +02:00
|
|
|
* Invoice Ninja (https://invoiceninja.com).
|
2020-02-05 05:06:03 +01:00
|
|
|
*
|
|
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
|
|
*
|
2024-04-12 06:15:41 +02:00
|
|
|
* @copyright Copyright (c) 2024. Invoice Ninja LLC (https://invoiceninja.com)
|
2020-02-05 05:06:03 +01:00
|
|
|
*
|
2021-06-16 08:58:16 +02:00
|
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
2020-02-05 05:06:03 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Http\Controllers;
|
|
|
|
|
2023-10-26 04:57:44 +02:00
|
|
|
use App\Exceptions\FilePermissionsFailure;
|
2023-09-18 03:20:01 +02:00
|
|
|
use App\Models\Company;
|
2023-10-26 04:57:44 +02:00
|
|
|
use App\Utils\Ninja;
|
2022-05-15 01:03:42 +02:00
|
|
|
use App\Utils\Traits\AppSetup;
|
2023-10-26 04:57:44 +02:00
|
|
|
use App\Utils\Traits\ClientGroupSettingsSaver;
|
|
|
|
use Illuminate\Foundation\Bus\DispatchesJobs;
|
2020-04-11 13:48:38 +02:00
|
|
|
use Illuminate\Support\Facades\Artisan;
|
2022-03-19 04:13:29 +01:00
|
|
|
use Illuminate\Support\Facades\Storage;
|
2022-03-19 04:16:45 +01:00
|
|
|
|
2020-02-05 05:06:03 +01:00
|
|
|
class SelfUpdateController extends BaseController
|
|
|
|
{
|
|
|
|
use DispatchesJobs;
|
2022-04-24 10:51:41 +02:00
|
|
|
use ClientGroupSettingsSaver;
|
2022-05-15 01:03:42 +02:00
|
|
|
use AppSetup;
|
2020-02-05 05:06:03 +01:00
|
|
|
|
2023-05-02 16:33:58 +02:00
|
|
|
private string $filename = 'invoiceninja.tar';
|
2023-05-02 12:33:48 +02:00
|
|
|
|
2022-04-14 13:51:25 +02:00
|
|
|
private array $purge_file_list = [
|
|
|
|
'bootstrap/cache/compiled.php',
|
|
|
|
'bootstrap/cache/config.php',
|
|
|
|
'bootstrap/cache/packages.php',
|
|
|
|
'bootstrap/cache/services.php',
|
|
|
|
'bootstrap/cache/routes-v7.php',
|
|
|
|
'bootstrap/cache/livewire-components.php',
|
|
|
|
];
|
|
|
|
|
2020-02-05 05:06:03 +01:00
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-03-19 04:10:00 +01:00
|
|
|
public function update()
|
2022-03-19 03:36:47 +01:00
|
|
|
{
|
2022-03-19 04:10:00 +01:00
|
|
|
set_time_limit(0);
|
|
|
|
define('STDIN', fopen('php://stdin', 'r'));
|
|
|
|
|
|
|
|
if (Ninja::isHosted()) {
|
|
|
|
return response()->json(['message' => ctrans('texts.self_update_not_available')], 403);
|
|
|
|
}
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Test filesystem is writable');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2023-08-17 02:57:06 +02:00
|
|
|
$this->testWritable();
|
2022-06-21 11:57:17 +02:00
|
|
|
|
|
|
|
nlog('Clear cache directory');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
|
|
|
$this->clearCacheDir();
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('copying release file');
|
2022-03-19 04:10:00 +01:00
|
|
|
|
2023-08-17 02:57:06 +02:00
|
|
|
$file_headers = @get_headers($this->getDownloadUrl());
|
|
|
|
|
2024-06-14 09:09:44 +02:00
|
|
|
if(!is_array($file_headers)) {
|
2024-04-28 00:11:51 +02:00
|
|
|
return response()->json(['message' => 'There was a problem reaching the update server, please try again in a little while.'], 410);
|
2024-06-14 09:09:44 +02:00
|
|
|
}
|
2024-04-28 00:11:51 +02:00
|
|
|
|
2024-01-14 05:05:00 +01:00
|
|
|
if (stripos($file_headers[0], "404 Not Found") > 0 || (stripos($file_headers[0], "302 Found") > 0 && stripos($file_headers[7], "404 Not Found") > 0)) {
|
2022-05-14 08:52:31 +02:00
|
|
|
return response()->json(['message' => 'Download not yet available. Please try again shortly.'], 410);
|
2022-06-21 11:57:17 +02:00
|
|
|
}
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2023-08-17 02:57:06 +02:00
|
|
|
try {
|
|
|
|
if (copy($this->getDownloadUrl(), storage_path("app/{$this->filename}"))) {
|
|
|
|
nlog('Copied file from URL');
|
|
|
|
}
|
2023-10-26 04:57:44 +02:00
|
|
|
} catch(\Exception $e) {
|
2023-08-17 02:57:06 +02:00
|
|
|
nlog($e->getMessage());
|
|
|
|
return response()->json(['message' => 'File exists on the server, however there was a problem downloading and copying to the local filesystem'], 500);
|
|
|
|
}
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Finished copying');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2023-06-12 12:48:23 +02:00
|
|
|
$file = Storage::disk('local')->path($this->filename);
|
2023-05-02 12:23:29 +02:00
|
|
|
|
2023-06-12 12:48:23 +02:00
|
|
|
nlog('Extracting tar');
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2023-06-12 12:48:23 +02:00
|
|
|
$phar = new \PharData($file);
|
|
|
|
$phar->extractTo(base_path(), null, true);
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2023-06-12 12:48:23 +02:00
|
|
|
nlog('Finished extracting files');
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2023-06-12 12:48:23 +02:00
|
|
|
unlink($file);
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Deleted release zip file');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
foreach ($this->purge_file_list as $purge_file_path) {
|
2022-04-14 13:51:25 +02:00
|
|
|
$purge_file = base_path($purge_file_path);
|
2022-06-21 11:57:17 +02:00
|
|
|
if (file_exists($purge_file)) {
|
|
|
|
unlink($purge_file);
|
|
|
|
}
|
2022-04-14 13:51:25 +02:00
|
|
|
}
|
2022-03-19 04:10:00 +01:00
|
|
|
|
2024-06-14 09:09:44 +02:00
|
|
|
if(Storage::disk('base')->directoryExists('resources/lang')) {
|
2024-04-29 08:50:04 +02:00
|
|
|
Storage::disk('base')->deleteDirectory('resources/lang');
|
2024-06-14 09:09:44 +02:00
|
|
|
}
|
2024-04-29 08:50:04 +02:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Removing cache files');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2022-03-19 04:10:00 +01:00
|
|
|
Artisan::call('clear-compiled');
|
|
|
|
Artisan::call('route:clear');
|
|
|
|
Artisan::call('view:clear');
|
|
|
|
Artisan::call('migrate', ['--force' => true]);
|
2023-04-06 03:07:24 +02:00
|
|
|
Artisan::call('config:clear');
|
2022-03-19 04:10:00 +01:00
|
|
|
|
2022-05-15 01:03:42 +02:00
|
|
|
$this->buildCache(true);
|
2022-03-19 04:10:00 +01:00
|
|
|
|
2023-09-18 03:20:01 +02:00
|
|
|
$this->runModelChecks();
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Called Artisan commands');
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
return response()->json(['message' => 'Update completed'], 200);
|
2022-06-14 01:01:25 +02:00
|
|
|
}
|
|
|
|
|
2023-09-18 03:20:01 +02:00
|
|
|
private function runModelChecks()
|
|
|
|
{
|
|
|
|
Company::query()
|
|
|
|
->cursor()
|
2023-10-26 04:57:44 +02:00
|
|
|
->each(function ($company) {
|
2023-09-18 03:20:01 +02:00
|
|
|
|
2023-10-26 04:57:44 +02:00
|
|
|
$settings = $company->settings;
|
2023-09-18 03:20:01 +02:00
|
|
|
|
2023-10-26 04:57:44 +02:00
|
|
|
if(property_exists($settings->pdf_variables, 'purchase_order_details')) {
|
|
|
|
return;
|
|
|
|
}
|
2023-09-18 03:20:01 +02:00
|
|
|
|
2023-10-26 04:57:44 +02:00
|
|
|
$pdf_variables = $settings->pdf_variables;
|
|
|
|
$pdf_variables->purchase_order_details = [];
|
|
|
|
$settings->pdf_variables = $pdf_variables;
|
|
|
|
$company->settings = $settings;
|
|
|
|
$company->save();
|
2023-09-18 03:20:01 +02:00
|
|
|
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-04-27 01:43:17 +02:00
|
|
|
private function clearCacheDir()
|
|
|
|
{
|
|
|
|
$directoryIterator = new \RecursiveDirectoryIterator(base_path('bootstrap/cache'), \RecursiveDirectoryIterator::SKIP_DOTS);
|
|
|
|
|
|
|
|
foreach (new \RecursiveIteratorIterator($directoryIterator) as $file) {
|
|
|
|
unlink(base_path('bootstrap/cache/').$file->getFileName());
|
2023-06-12 12:48:23 +02:00
|
|
|
$file = null;
|
2022-04-27 01:43:17 +02:00
|
|
|
}
|
2023-04-21 09:38:05 +02:00
|
|
|
|
|
|
|
$directoryIterator = null;
|
2022-04-27 01:43:17 +02:00
|
|
|
}
|
|
|
|
|
2021-04-12 06:36:51 +02:00
|
|
|
private function testWritable()
|
|
|
|
{
|
2021-04-19 02:54:16 +02:00
|
|
|
$directoryIterator = new \RecursiveDirectoryIterator(base_path(), \RecursiveDirectoryIterator::SKIP_DOTS);
|
2021-04-12 06:36:51 +02:00
|
|
|
|
|
|
|
foreach (new \RecursiveIteratorIterator($directoryIterator) as $file) {
|
2022-06-21 11:57:17 +02:00
|
|
|
if (strpos($file->getPathname(), '.git') !== false) {
|
2021-04-19 02:54:16 +02:00
|
|
|
continue;
|
2022-06-21 11:57:17 +02:00
|
|
|
}
|
2021-04-19 02:54:16 +02:00
|
|
|
|
2021-04-12 06:36:51 +02:00
|
|
|
if ($file->isFile() && ! $file->isWritable()) {
|
2023-05-02 12:23:29 +02:00
|
|
|
|
2021-04-30 06:29:27 +02:00
|
|
|
nlog("Cannot update system because {$file->getFileName()} is not writable");
|
2021-04-30 06:22:36 +02:00
|
|
|
throw new FilePermissionsFailure("Cannot update system because {$file->getFileName()} is not writable");
|
2022-06-21 11:57:17 +02:00
|
|
|
|
2021-04-12 06:36:51 +02:00
|
|
|
}
|
2023-06-12 12:48:23 +02:00
|
|
|
|
|
|
|
$file = null;
|
2021-04-12 06:36:51 +02:00
|
|
|
}
|
|
|
|
|
2023-04-21 09:38:05 +02:00
|
|
|
$directoryIterator = null;
|
2024-01-14 05:05:00 +01:00
|
|
|
|
2021-04-12 06:36:51 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-11-11 01:13:39 +01:00
|
|
|
public function checkVersion()
|
|
|
|
{
|
|
|
|
return trim(file_get_contents(config('ninja.version_url')));
|
|
|
|
}
|
2022-03-19 03:00:29 +01:00
|
|
|
|
|
|
|
private function getDownloadUrl()
|
|
|
|
{
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2023-06-12 12:48:23 +02:00
|
|
|
$version = $this->checkVersion();
|
2023-05-02 12:33:48 +02:00
|
|
|
|
2023-05-02 16:33:58 +02:00
|
|
|
return "https://github.com/invoiceninja/invoiceninja/releases/download/v{$version}/invoiceninja.tar";
|
2023-05-02 12:33:48 +02:00
|
|
|
|
2022-03-19 03:00:29 +01:00
|
|
|
}
|
2020-03-21 06:37:30 +01:00
|
|
|
}
|