mirror of
https://github.com/InvoiceShelf/InvoiceShelf.git
synced 2026-04-07 21:44:51 +00:00
* Convert string references to `::class` PHP 5.5.9 adds the new static `class` property which provides the fully qualified class name. This is preferred over using strings for class names since the `class` property references are checked by PHP. * Use Faker methods Accessing Faker properties was deprecated in Faker 1.14. * Convert route options to fluent methods Laravel 8 adopts the tuple syntax for controller actions. Since the old options array is incompatible with this syntax, Shift converted them to use modern, fluent methods. * Adopt class based routes * Remove default `app` files * Shift core files * Streamline config files * Set new `ENV` variables * Default new `bootstrap/app.php` * Re-register HTTP middleware * Consolidate service providers * Re-register service providers * Re-register routes * Re-register scheduled commands * Bump Composer dependencies * Use `<env>` tags for configuration `<env>` tags have a lower precedence than system environment variables making it easier to overwrite PHPUnit configuration values in additional environments, such a CI. Review this blog post for more details on configuration precedence when testing Laravel: https://jasonmccreary.me/articles/laravel-testing-configuration-precedence/ * Adopt anonymous migrations * Rename `password_resets` table * Convert `$casts` property to method * Adopt Laravel type hints * Mark base controller as `abstract` * Remove `CreatesApplication` testing trait * Shift cleanup * Fix shift first issues * Updating Rules for laravel 11, sanctum config and pint * Fix Carbon issue on dashboard * Temporary fix for tests while migration is issue fixed on laravel side * Carbon needs numerical values, not strings * Minimum php version * Fix domain installation step not fetching the correct company_id * Fix Role Policy wasn't properly registered ---------
232 lines
6.4 KiB
PHP
232 lines
6.4 KiB
PHP
<?php
|
|
|
|
namespace App\Space;
|
|
|
|
use App\Events\ModuleEnabledEvent;
|
|
use App\Events\ModuleInstalledEvent;
|
|
use App\Http\Resources\ModuleResource;
|
|
use App\Models\Module as ModelsModule;
|
|
use App\Models\Setting;
|
|
use Artisan;
|
|
use File;
|
|
use GuzzleHttp\Exception\RequestException;
|
|
use Nwidart\Modules\Facades\Module;
|
|
use ZipArchive;
|
|
|
|
// Implementation taken from Akaunting - https://github.com/akaunting/akaunting
|
|
class ModuleInstaller
|
|
{
|
|
use SiteApi;
|
|
|
|
public static function getModules()
|
|
{
|
|
$data = null;
|
|
if (env('APP_ENV') === 'development') {
|
|
$url = 'api/marketplace/modules?is_dev=1';
|
|
} else {
|
|
$url = 'api/marketplace/modules';
|
|
}
|
|
|
|
$token = Setting::getSetting('api_token');
|
|
$response = static::getRemote($url, ['timeout' => 100, 'track_redirects' => true], $token);
|
|
|
|
if ($response && ($response->getStatusCode() == 401)) {
|
|
return response()->json(['error' => 'invalid_token']);
|
|
}
|
|
|
|
if ($response && ($response->getStatusCode() == 200)) {
|
|
$data = $response->getBody()->getContents();
|
|
}
|
|
|
|
$data = json_decode($data);
|
|
|
|
return ModuleResource::collection(collect($data->modules));
|
|
}
|
|
|
|
public static function getModule($module)
|
|
{
|
|
$data = null;
|
|
if (env('APP_ENV') === 'development') {
|
|
$url = 'api/marketplace/modules/'.$module.'?is_dev=1';
|
|
} else {
|
|
$url = 'api/marketplace/modules/'.$module;
|
|
}
|
|
|
|
$token = Setting::getSetting('api_token');
|
|
$response = static::getRemote($url, ['timeout' => 100, 'track_redirects' => true], $token);
|
|
|
|
if ($response && ($response->getStatusCode() == 401)) {
|
|
return (object) ['success' => false, 'error' => 'invalid_token'];
|
|
}
|
|
|
|
if ($response && ($response->getStatusCode() == 200)) {
|
|
$data = $response->getBody()->getContents();
|
|
}
|
|
|
|
$data = json_decode($data);
|
|
|
|
return $data;
|
|
}
|
|
|
|
public static function upload($request)
|
|
{
|
|
// Create temp directory
|
|
$temp_dir = storage_path('app/temp-'.md5(mt_rand()));
|
|
|
|
if (! File::isDirectory($temp_dir)) {
|
|
File::makeDirectory($temp_dir);
|
|
}
|
|
|
|
$path = $request->file('avatar')->storeAs(
|
|
'temp-'.md5(mt_rand()),
|
|
$request->module.'.zip',
|
|
'local'
|
|
);
|
|
|
|
return $path;
|
|
}
|
|
|
|
public static function download($module, $version)
|
|
{
|
|
$data = null;
|
|
$path = null;
|
|
|
|
if (env('APP_ENV') === 'development') {
|
|
$url = "api/marketplace/modules/file/{$module}?version={$version}&is_dev=1";
|
|
} else {
|
|
$url = "api/marketplace/modules/file/{$module}?version={$version}";
|
|
}
|
|
|
|
$token = Setting::getSetting('api_token');
|
|
$response = static::getRemote($url, ['timeout' => 100, 'track_redirects' => true], $token);
|
|
|
|
// Exception
|
|
if ($response instanceof RequestException) {
|
|
return [
|
|
'success' => false,
|
|
'error' => 'Download Exception',
|
|
'data' => [
|
|
'path' => $path,
|
|
],
|
|
];
|
|
}
|
|
|
|
if ($response && ($response->getStatusCode() == 401 || $response->getStatusCode() == 404 || $response->getStatusCode() == 500)) {
|
|
return json_decode($response->getBody()->getContents());
|
|
}
|
|
|
|
if ($response && ($response->getStatusCode() == 200)) {
|
|
$data = $response->getBody()->getContents();
|
|
}
|
|
|
|
// Create temp directory
|
|
$temp_dir = storage_path('app/temp-'.md5(mt_rand()));
|
|
|
|
if (! File::isDirectory($temp_dir)) {
|
|
File::makeDirectory($temp_dir);
|
|
}
|
|
|
|
$zip_file_path = $temp_dir.'/upload.zip';
|
|
|
|
// Add content to the Zip file
|
|
$uploaded = is_int(file_put_contents($zip_file_path, $data)) ? true : false;
|
|
|
|
if (! $uploaded) {
|
|
return false;
|
|
}
|
|
|
|
return [
|
|
'success' => true,
|
|
'path' => $zip_file_path,
|
|
];
|
|
}
|
|
|
|
public static function unzip($module, $zip_file_path)
|
|
{
|
|
if (! file_exists($zip_file_path)) {
|
|
throw new \Exception('Zip file not found');
|
|
}
|
|
|
|
$temp_extract_dir = storage_path('app/temp2-'.md5(mt_rand()));
|
|
|
|
if (! File::isDirectory($temp_extract_dir)) {
|
|
File::makeDirectory($temp_extract_dir);
|
|
}
|
|
// Unzip the file
|
|
$zip = new ZipArchive();
|
|
|
|
if ($zip->open($zip_file_path)) {
|
|
$zip->extractTo($temp_extract_dir);
|
|
}
|
|
|
|
$zip->close();
|
|
|
|
// Delete zip file
|
|
File::delete($zip_file_path);
|
|
|
|
return $temp_extract_dir;
|
|
}
|
|
|
|
public static function copyFiles($module, $temp_extract_dir)
|
|
{
|
|
if (! File::isDirectory(base_path('Modules'))) {
|
|
File::makeDirectory(base_path('Modules'));
|
|
}
|
|
|
|
// Delete Existing Module directory
|
|
if (! File::isDirectory(base_path('Modules').'/'.$module)) {
|
|
File::deleteDirectory(base_path('Modules').'/'.$module);
|
|
}
|
|
|
|
if (! File::copyDirectory($temp_extract_dir, base_path('Modules').'/')) {
|
|
return false;
|
|
}
|
|
|
|
// Delete temp directory
|
|
File::deleteDirectory($temp_extract_dir);
|
|
|
|
return true;
|
|
}
|
|
|
|
public static function deleteFiles($json)
|
|
{
|
|
$files = json_decode($json);
|
|
|
|
foreach ($files as $file) {
|
|
\File::delete(base_path($file));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public static function complete($module, $version)
|
|
{
|
|
Module::register();
|
|
|
|
Artisan::call("module:migrate $module --force");
|
|
Artisan::call("module:seed $module --force");
|
|
Artisan::call("module:enable $module");
|
|
|
|
$module = ModelsModule::updateOrCreate(['name' => $module], ['version' => $version, 'installed' => true, 'enabled' => true]);
|
|
|
|
ModuleInstalledEvent::dispatch($module);
|
|
ModuleEnabledEvent::dispatch($module);
|
|
|
|
return true;
|
|
}
|
|
|
|
public static function checkToken(string $token)
|
|
{
|
|
$url = 'api/marketplace/ping';
|
|
$response = static::getRemote($url, ['timeout' => 100, 'track_redirects' => true], $token);
|
|
|
|
if ($response && ($response->getStatusCode() == 200)) {
|
|
$data = $response->getBody()->getContents();
|
|
|
|
return response()->json(json_decode($data));
|
|
}
|
|
|
|
return response()->json(['error' => 'invalid_token']);
|
|
}
|
|
}
|