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
|
|
|
|
*
|
2022-04-27 05:20:41 +02:00
|
|
|
* @copyright Copyright (c) 2022. 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;
|
|
|
|
|
2021-04-12 06:36:51 +02:00
|
|
|
use App\Exceptions\FilePermissionsFailure;
|
2022-04-24 10:51:41 +02:00
|
|
|
use App\Models\Client;
|
2020-03-21 06:37:30 +01:00
|
|
|
use App\Utils\Ninja;
|
2022-05-15 01:03:42 +02:00
|
|
|
use App\Utils\Traits\AppSetup;
|
2022-04-24 10:51:41 +02:00
|
|
|
use App\Utils\Traits\ClientGroupSettingsSaver;
|
2022-06-01 00:59:00 +02:00
|
|
|
use Beganovich\Snappdf\Snappdf;
|
2020-02-10 10:53:02 +01:00
|
|
|
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
|
|
|
|
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()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-02-12 11:06:59 +01:00
|
|
|
/**
|
|
|
|
* @OA\Post(
|
|
|
|
* path="/api/v1/self-update",
|
|
|
|
* operationId="selfUpdate",
|
|
|
|
* tags={"update"},
|
|
|
|
* summary="Performs a system update",
|
|
|
|
* description="Performs a system update",
|
|
|
|
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
|
|
|
|
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
|
|
|
|
* @OA\Parameter(ref="#/components/parameters/X-Api-Password"),
|
|
|
|
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
|
|
|
|
* @OA\Parameter(ref="#/components/parameters/include"),
|
|
|
|
* @OA\Response(
|
|
|
|
* response=200,
|
|
|
|
* description="Success/failure response"
|
|
|
|
* ),
|
|
|
|
* @OA\Response(
|
|
|
|
* response=422,
|
|
|
|
* description="Validation error",
|
|
|
|
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
|
|
|
|
*
|
|
|
|
* ),
|
|
|
|
* @OA\Response(
|
|
|
|
* response="default",
|
|
|
|
* description="Unexpected Error",
|
|
|
|
* @OA\JsonContent(ref="#/components/schemas/Error"),
|
|
|
|
* ),
|
|
|
|
* )
|
|
|
|
*/
|
2022-09-27 12:01:58 +02:00
|
|
|
|
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
|
|
|
|
2022-03-19 04:10:00 +01: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
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
if (copy($this->getDownloadUrl(), storage_path('app/invoiceninja.zip'))) {
|
|
|
|
nlog('Copied file from URL');
|
|
|
|
} else {
|
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
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Finished copying');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2022-03-19 03:36:47 +01:00
|
|
|
$file = Storage::disk('local')->path('invoiceninja.zip');
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Extracting zip');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2022-06-14 01:01:25 +02:00
|
|
|
//clean up old snappdf installations
|
2022-10-09 09:32:53 +02:00
|
|
|
//$this->cleanOldSnapChromeBinaries();
|
2022-06-21 11:57:17 +02:00
|
|
|
|
2022-06-02 08:42:37 +02:00
|
|
|
$zipFile = new \PhpZip\ZipFile();
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2022-06-02 08:42:37 +02:00
|
|
|
$zipFile->openFile($file);
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2022-06-02 08:42:37 +02:00
|
|
|
$zipFile->extractTo(base_path());
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2022-06-02 08:42:37 +02:00
|
|
|
$zipFile->close();
|
2022-05-25 12:53:12 +02:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
nlog('Finished extracting files');
|
2022-05-06 00:40:34 +02:00
|
|
|
|
2022-03-19 03:36:47 +01:00
|
|
|
unlink($file);
|
|
|
|
|
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
|
|
|
|
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]);
|
|
|
|
Artisan::call('optimize');
|
|
|
|
|
2022-05-15 01:03:42 +02:00
|
|
|
$this->buildCache(true);
|
2022-03-19 04:10:00 +01:00
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
private function cleanOldSnapChromeBinaries()
|
|
|
|
{
|
2022-06-21 11:57:17 +02:00
|
|
|
$current_revision = base_path('vendor/beganovich/snappdf/versions/revision.txt');
|
2022-06-16 05:04:05 +02:00
|
|
|
$current_revision_text = file_get_contents($current_revision);
|
2022-06-14 01:01:25 +02:00
|
|
|
|
2022-06-16 05:04:05 +02:00
|
|
|
$iterator = new \DirectoryIterator(base_path('vendor/beganovich/snappdf/versions'));
|
2022-06-14 01:01:25 +02:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
foreach ($iterator as $file) {
|
|
|
|
if ($file->isDir() && ! $file->isDot() && ($current_revision_text != $file->getFileName())) {
|
2022-06-16 05:07:53 +02:00
|
|
|
$directoryIterator = new \RecursiveDirectoryIterator(base_path('vendor/beganovich/snappdf/versions/'.$file->getFileName()), \RecursiveDirectoryIterator::SKIP_DOTS);
|
2022-06-14 01:01:25 +02:00
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
foreach (new \RecursiveIteratorIterator($directoryIterator) as $filex) {
|
|
|
|
unlink($filex->getPathName());
|
2022-06-16 05:07:53 +02:00
|
|
|
}
|
2022-06-14 01:01:25 +02:00
|
|
|
|
2022-06-16 05:36:38 +02:00
|
|
|
$this->deleteDirectory(base_path('vendor/beganovich/snappdf/versions/'.$file->getFileName()));
|
2022-06-16 05:04:05 +02:00
|
|
|
}
|
|
|
|
}
|
2022-03-19 03:36:47 +01:00
|
|
|
}
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
private function deleteDirectory($dir)
|
|
|
|
{
|
|
|
|
if (! file_exists($dir)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! is_dir($dir) || is_link($dir)) {
|
|
|
|
return unlink($dir);
|
|
|
|
}
|
|
|
|
foreach (scandir($dir) as $item) {
|
|
|
|
if ($item == '.' || $item == '..') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (! $this->deleteDirectory($dir.'/'.$item)) {
|
|
|
|
if (! $this->deleteDirectory($dir.'/'.$item)) {
|
|
|
|
return false;
|
|
|
|
}
|
2022-06-16 05:36:38 +02:00
|
|
|
}
|
2022-06-21 11:57:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return rmdir($dir);
|
2022-06-16 05:36:38 +02:00
|
|
|
}
|
|
|
|
|
2022-04-24 10:51:41 +02:00
|
|
|
private function postHookUpdate()
|
|
|
|
{
|
2022-06-21 11:57:17 +02:00
|
|
|
if (config('ninja.app_version') == '5.3.82') {
|
|
|
|
Client::withTrashed()->cursor()->each(function ($client) {
|
2022-04-24 10:51:41 +02:00
|
|
|
$entity_settings = $this->checkSettingType($client->settings);
|
|
|
|
$entity_settings->md5 = md5(time());
|
|
|
|
$client->settings = $entity_settings;
|
|
|
|
$client->save();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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()) {
|
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
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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()
|
|
|
|
{
|
|
|
|
$version = $this->checkVersion();
|
2022-03-19 03:36:47 +01:00
|
|
|
|
2022-03-19 04:13:29 +01:00
|
|
|
return "https://github.com/invoiceninja/invoiceninja/releases/download/v{$version}/invoiceninja.zip";
|
2022-03-19 03:00:29 +01:00
|
|
|
}
|
2020-03-21 06:37:30 +01:00
|
|
|
}
|