1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-14 07:02:34 +01:00
invoiceninja/app/Http/Controllers/SelfUpdateController.php

271 lines
8.4 KiB
PHP
Raw Normal View History

<?php
/**
* Invoice Ninja (https://invoiceninja.com).
*
* @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)
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
*/
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;
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;
use Beganovich\Snappdf\Snappdf;
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;
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()
{
}
/**
* @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-03-19 04:10:00 +01:00
// public function old_update(\Codedge\Updater\UpdaterManager $updater)
// {
// set_time_limit(0);
// define('STDIN', fopen('php://stdin', 'r'));
2020-06-04 13:42:45 +02:00
2022-03-19 04:10:00 +01:00
// if (Ninja::isHosted()) {
// return response()->json(['message' => ctrans('texts.self_update_not_available')], 403);
// }
2022-03-19 04:10:00 +01:00
// $this->testWritable();
2021-04-12 06:36:51 +02:00
2022-03-19 04:10:00 +01:00
// // Get the new version available
// $versionAvailable = $updater->source()->getVersionAvailable();
2021-04-09 02:33:54 +02:00
2022-03-19 04:10:00 +01:00
// // Create a release
// $release = $updater->source()->fetch($versionAvailable);
2021-04-09 02:33:54 +02:00
2022-03-19 04:10:00 +01:00
// $updater->source()->update($release);
2021-04-09 02:33:54 +02:00
2022-03-19 04:10:00 +01:00
// $cacheCompiled = base_path('bootstrap/cache/compiled.php');
// if (file_exists($cacheCompiled)) { unlink ($cacheCompiled); }
// $cacheServices = base_path('bootstrap/cache/services.php');
// if (file_exists($cacheServices)) { unlink ($cacheServices); }
2021-04-09 07:03:06 +02:00
2022-03-19 04:10:00 +01:00
// Artisan::call('clear-compiled');
// Artisan::call('route:clear');
// Artisan::call('view:clear');
// Artisan::call('optimize');
2022-03-19 04:10:00 +01:00
// return response()->json(['message' => 'Update completed'], 200);
2022-03-19 04:10:00 +01:00
// }
2020-11-11 01:13:39 +01: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);
}
nlog('Test filesystem is writable');
2022-05-06 00:40:34 +02:00
2022-03-19 04:10:00 +01: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
if (copy($this->getDownloadUrl(), storage_path('app/invoiceninja.zip'))) {
nlog('Copied file from URL');
} else {
return response()->json(['message' => 'Download not yet available. Please try again shortly.'], 410);
}
2022-03-19 03:36:47 +01: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');
nlog('Extracting zip');
2022-05-06 00:40:34 +02:00
//clean up old snappdf installations
$this->cleanOldSnapChromeBinaries();
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
nlog('Finished extracting files');
2022-05-06 00:40:34 +02:00
2022-03-19 03:36:47 +01:00
unlink($file);
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('optimize');
2022-05-15 01:03:42 +02:00
$this->buildCache(true);
2022-03-19 04:10:00 +01:00
nlog('Called Artisan commands');
2022-03-19 03:36:47 +01:00
return response()->json(['message' => 'Update completed'], 200);
}
private function cleanOldSnapChromeBinaries()
{
$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-16 05:04:05 +02:00
$iterator = new \DirectoryIterator(base_path('vendor/beganovich/snappdf/versions'));
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);
foreach (new \RecursiveIteratorIterator($directoryIterator) as $filex) {
unlink($filex->getPathName());
2022-06-16 05:07:53 +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
}
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
}
}
return rmdir($dir);
2022-06-16 05:36:38 +02:00
}
2022-04-24 10:51:41 +02:00
private function postHookUpdate()
{
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) {
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()) {
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
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
}
}