1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-16 16:13:20 +01:00
invoiceninja/app/Http/Controllers/SelfUpdateController.php

190 lines
5.4 KiB
PHP
Raw Normal View History

<?php
/**
* Invoice Ninja (https://invoiceninja.com).
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
2023-01-28 23:21:40 +01:00
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
*/
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
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;
2023-05-02 16:33:58 +02:00
// private bool $use_zip = false;
2023-05-02 12:33:48 +02: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',
];
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);
}
nlog('Test filesystem is writable');
2022-05-06 00:40:34 +02:00
2023-08-17 02:57:06 +02:00
$this->testWritable();
nlog('Clear cache directory');
2022-05-06 00:40:34 +02:00
$this->clearCacheDir();
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());
if (stripos($file_headers[0], "404 Not Found") >0 || (stripos($file_headers[0], "302 Found") > 0 && stripos($file_headers[7], "404 Not Found") > 0)) {
return response()->json(['message' => 'Download not yet available. Please try again shortly.'], 410);
}
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);
}
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
nlog('Deleted release zip file');
2022-05-06 00:40:34 +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);
if (file_exists($purge_file)) {
unlink($purge_file);
}
2022-04-14 13:51:25 +02:00
}
2022-03-19 04:10:00 +01: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]);
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();
nlog('Called Artisan commands');
2022-03-19 03:36:47 +01:00
return response()->json(['message' => 'Update completed'], 200);
}
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) {
if (strpos($file->getPathname(), '.git') !== false) {
2021-04-19 02:54:16 +02:00
continue;
}
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");
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;
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
}
}