plugin list works now finally

This commit is contained in:
2025-08-09 16:43:19 +02:00
parent 543127d339
commit b1de8f53c6
16 changed files with 485 additions and 251 deletions

4
PRP.md
View File

@@ -15,9 +15,9 @@ This document outlines the architecture, functionality, and implementation detai
## 2. Core Technologies & Stack
* **Languages:** PHP 8.1, JavaScript
* **Languages:** PHP 8.3, JavaScript
* **Frameworks & Runtimes:** Laravel 12.21.0, Vue.js 3.5.18, Inertia.js 1.3.0, Livewire 3.6.4, Vite 5.4.19
* **Databases:** MySQL (default), with configurations for PostgreSQL, SQLite, and SQL Server. Redis is used for caching.
* **Databases:** SQLite. Redis is used for caching.
* **Key PHP Libraries/Dependencies:**
* Filament 3.3.34 (for the admin panel)
* Guzzle 7.9.3 (for HTTP requests)

View File

@@ -1,133 +0,0 @@
<?php
namespace App\Filament\Pages;
use App\Api\Plugins\ApiPluginInterface;
use App\Filament\Resources\PluginResource\Plugin;
use Filament\Notifications\Notification;
use Filament\Pages\Page;
use Filament\Tables\Columns\IconColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Concerns\InteractsWithTable;
use Filament\Tables\Contracts\HasTable;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Support\Facades\File;
use App\Models\ApiProvider;
use Filament\Tables\Actions\Action;
class ListPlugins extends Page implements HasTable
{
use InteractsWithTable;
protected static ?string $navigationIcon = 'heroicon-o-puzzle-piece';
protected static string $view = 'filament.pages.list-plugins';
protected static ?string $navigationGroup = 'Plugins';
protected static ?string $title = 'Plugins';
public function table(Table $table): Table
{
$plugins = new Collection();
$path = app_path('Api/Plugins');
if (File::exists($path)) {
$files = File::files($path);
foreach ($files as $file) {
$filename = $file->getFilenameWithoutExtension();
$class = 'App\\Api\\Plugins\\' . $filename;
if (class_exists($class) && in_array(ApiPluginInterface::class, class_implements($class))) {
try {
$apiProvider = ApiProvider::where('plugin', $filename)->first();
if(!$apiProvider) continue;
$instance = new $class($apiProvider);
$plugins->add(new Plugin([
'id' => $instance->getIdentifier(),
'name' => $instance->getName(),
'identifier' => $instance->getIdentifier(),
'enabled' => $instance->isEnabled(),
'file_path' => $file->getPathname(),
]));
} catch (\Exception $e) {
// Log error or handle gracefully if a plugin cannot be instantiated
// For now, we'll just skip it
}
}
}
}
return $table
->query(Plugin::query()->setCollection($plugins))
->columns([
TextColumn::make('name')
->label('Name')
->searchable()
->sortable(),
TextColumn::make('identifier')
->label('Identifier'),
IconColumn::make('enabled')
->label('Enabled')
->boolean(),
TextColumn::make('file_path')
->label('File Path')
->toggleable(isToggledHiddenByDefault: true),
])
->actions([
Action::make('toggle_enabled')
->label(fn ($record) => $record->enabled ? 'Disable' : 'Enable')
->icon(fn ($record) => $record->enabled ? 'heroicon-o-x-circle' : 'heroicon-o-check-circle')
->action(function ($record) {
try {
$apiProvider = ApiProvider::where('plugin', $record->identifier)->first();
if(!$apiProvider) throw new \Exception('ApiProvider not found');
$pluginClass = 'App\\Api\\Plugins\\' . $record->identifier;
$plugin = new $pluginClass($apiProvider);
if ($record->enabled) {
$plugin->disable();
} else {
$plugin->enable();
}
Notification::make()
->title('Plugin status updated')
->success()
->send();
} catch (\Exception $e) {
Notification::make()
->title('Error updating plugin status')
->body($e->getMessage())
->danger()
->send();
}
}),
Action::make('delete')
->label('Delete')
->icon('heroicon-o-trash')
->color('danger')
->requiresConfirmation()
->action(function ($record) {
try {
File::delete($record->file_path);
Notification::make()
->title('Plugin deleted successfully')
->success()
->send();
} catch (\Exception $e) {
Notification::make()
->title('Error deleting plugin')
->body($e->getMessage())
->danger()
->send();
}
}),
])
->bulkActions([
// No bulk actions for now
]);
}
// Removed getTableRecords() as data is now provided via query()
}

View File

@@ -0,0 +1,95 @@
<?php
namespace App\Filament\Pages;
use Illuminate\Database\Eloquent\Model;
use App\Api\Plugins\ApiPluginInterface;
use App\Models\ApiProvider;
use Illuminate\Support\Facades\File;
use Exception;
use ReflectionClass;
class Plugin extends Model
{
protected $table = null; // No actual table
protected $guarded = []; // Allow mass assignment for all attributes
public $incrementing = false;
protected $keyType = 'string';
protected $primaryKey = 'id';
public function __construct(array $attributes = [])
{
parent::__construct($attributes);
foreach ($attributes as $key => $value) {
$this->$key = $value;
}
}
public static function getAllPlugins()
{
$plugins = [];
$path = app_path('Api/Plugins');
if (File::exists($path)) {
$files = File::files($path);
foreach ($files as $file) {
$filename = $file->getFilenameWithoutExtension();
if (in_array($filename, ['ApiPluginInterface', 'LoggablePlugin', 'PluginLoader'])) {
continue;
}
$class = 'App\Api\Plugins\\' . $filename;
if (class_exists($class) && in_array(ApiPluginInterface::class, class_implements($class))) {
try {
// Check if there's an ApiProvider for this plugin
$apiProvider = ApiProvider::where('plugin', $filename)->first();
$hasApiProvider = $apiProvider !== null;
// Get plugin information without instantiating the class
// This avoids issues with plugins requiring ApiProvider in constructor
$reflection = new ReflectionClass($class);
$instance = $reflection->newInstanceWithoutConstructor();
$plugins[] = new self([
'id' => $instance->getIdentifier(),
'name' => $instance->getName(),
'identifier' => $instance->getIdentifier(),
'enabled' => $hasApiProvider && $apiProvider->enabled,
'file_path' => $file->getPathname(),
'has_api_provider' => $hasApiProvider,
'configured' => $hasApiProvider
]);
} catch (Exception $e) {
// Log error or handle as needed
}
}
}
}
return $plugins;
}
public function newCollection(array $models = [])
{
return new \Illuminate\Database\Eloquent\Collection($models);
}
public function newQuery()
{
// Create a new query builder instance
$query = new \Illuminate\Database\Eloquent\Builder(
new \Illuminate\Database\Query\Builder(
\Illuminate\Support\Facades\DB::connection()->getQueryGrammar(),
\Illuminate\Support\Facades\DB::connection()->getPostProcessor()
)
);
// Set the model for the query builder
$query->setModel($this);
return $query;
}
}

View File

@@ -0,0 +1,62 @@
<?php
namespace App\Filament\Pages;
use Filament\Pages\Page;
use Filament\Tables;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Filament\Forms\Contracts\HasForms;
use Filament\Forms\Concerns\InteractsWithForms;
use App\Models\Plugin;
class Plugins extends Page implements Tables\Contracts\HasTable, HasForms
{
use Tables\Concerns\InteractsWithTable;
use InteractsWithForms;
protected static ?string $navigationIcon = 'heroicon-o-puzzle-piece';
protected static ?string $navigationGroup = 'Plugins';
protected static ?string $title = 'Plugins';
protected static string $view = 'filament.pages.plugins';
protected static ?string $slug = 'list-plugins';
public function table(Table $table): Table
{
return $table
->query(Plugin::query()) // Use a dummy query
->columns([
Tables\Columns\TextColumn::make('name')
->label('Name')
->searchable()
->sortable(),
Tables\Columns\TextColumn::make('identifier')
->label('Identifier'),
Tables\Columns\IconColumn::make('enabled')
->label('Enabled')
->boolean(),
Tables\Columns\IconColumn::make('configured')
->label('Configured')
->boolean()
->tooltip(fn ($record) => $record->configured ? 'Has ApiProvider record' : 'No ApiProvider record'),
Tables\Columns\TextColumn::make('file_path')
->label('File Path')
->toggleable(isToggledHiddenByDefault: true),
]);
}
public function getTableRecords(): \Illuminate\Database\Eloquent\Collection
{
// Get all plugins as a collection
return Plugin::getAllPlugins();
}
public function currentlyValidatingForm(\Filament\Forms\ComponentContainer|null $form): void
{
// No form validation needed for this page
}
}

View File

@@ -1,74 +0,0 @@
<?php
namespace App\Filament\Resources\PluginResource;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Pagination\LengthAwarePaginator;
use Illuminate\Pagination\Paginator;
class CollectionEloquentBuilder extends Builder
{
protected $collection;
public function __construct($query)
{
parent::__construct($query);
$this->collection = new Collection(); // Initialize with an empty collection
}
public function setCollection(Collection $collection)
{
$this->collection = $collection;
return $this;
}
public function get($columns = ['*'])
{
return $this->collection;
}
public function find($id, $columns = ['*'])
{
return $this->collection->firstWhere('id', $id);
}
public function paginate($perPage = null, $columns = ['*'], $pageName = 'page', $page = null, $total = null)
{
$page = $page ?: Paginator::resolveCurrentPage($pageName);
$results = $this->collection->slice(($page - 1) * $perPage, $perPage)->all();
return new LengthAwarePaginator($results, $this->collection->count(), $perPage, $page, [
'path' => Paginator::resolveCurrentPath(),
'pageName' => $pageName,
]);
}
public function count($columns = '*')
{
return $this->collection->count();
}
public function where($column, $operator = null, $value = null, $boolean = 'and')
{
if (func_num_args() === 2) {
[$value, $operator] = [$operator, '='];
}
if ($operator === '=') {
$this->collection = $this->collection->where($column, $value);
} else {
// For simplicity, only handling '=' operator for now. More complex operators would require more logic.
// For example, for 'like', you'd need to implement string matching.
}
return $this;
}
public function orderBy($column, $direction = 'asc')
{
$this->collection = $this->collection->sortBy($column, SORT_REGULAR, $direction === 'desc');
return $this;
}
}

View File

@@ -1,36 +0,0 @@
<?php
namespace App\Filament\Resources\PluginResource;
use Illuminate\Database\Eloquent\Model;
class Plugin extends Model
{
protected $table = null; // No actual table
protected $guarded = []; // Allow mass assignment for all attributes
public $incrementing = false;
protected $keyType = 'string';
protected $primaryKey = 'id';
public function __construct(array $attributes = [])
{
parent::__construct($attributes);
foreach ($attributes as $key => $value) {
$this->$key = $value;
}
}
public function newCollection(array $models = [])
{
return new \Illuminate\Database\Eloquent\Collection($models);
}
public function newEloquentBuilder($query)
{
return new CollectionEloquentBuilder($query);
}
}

View File

@@ -21,6 +21,8 @@ class SettingResource extends Resource
protected static ?string $navigationIcon = 'heroicon-o-cog';
protected static ?string $navigationGroup = 'Einstellungen';
public static function form(Form $form): Form
{
return $form

View File

@@ -0,0 +1,16 @@
<?php
namespace App\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use App\Models\Plugin;
use Illuminate\Http\Request;
class PluginController extends Controller
{
public function index()
{
$plugins = Plugin::getAllPlugins();
return view('admin.plugins.index', compact('plugins'));
}
}

View File

@@ -0,0 +1,121 @@
<?php
namespace App\Livewire;
use App\Api\Plugins\ApiPluginInterface;
use App\Models\Plugin;
use App\Models\ApiProvider;
use Filament\Forms\Concerns\InteractsWithForms;
use Filament\Forms\Contracts\HasForms;
use Filament\Notifications\Notification;
use Filament\Tables\Actions\Action;
use Filament\Tables\Columns\IconColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Concerns\InteractsWithTable;
use Filament\Tables\Contracts\HasTable;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Support\Facades\File;
use Illuminate\Support\Facades\Log;
use Livewire\Component;
use Exception;
class ListPlugins extends Component implements HasForms, HasTable
{
use InteractsWithForms;
use InteractsWithTable;
public function getPlugins(): Collection
{
$plugins = new Collection();
$path = app_path('Api/Plugins');
if (File::exists($path)) {
$files = File::files($path);
foreach ($files as $file) {
$filename = $file->getFilenameWithoutExtension();
if (in_array($filename, ['ApiPluginInterface', 'LoggablePlugin', 'PluginLoader'])) {
continue;
}
$class = 'App\Api\Plugins\\' . $filename;
if (class_exists($class) && in_array(ApiPluginInterface::class, class_implements($class))) {
try {
// Check if there's an ApiProvider for this plugin
$apiProvider = ApiProvider::where('plugin', $filename)->first();
$hasApiProvider = $apiProvider !== null;
// Get plugin information without instantiating the class
// This avoids issues with plugins requiring ApiProvider in constructor
$reflection = new \ReflectionClass($class);
$instance = $reflection->newInstanceWithoutConstructor();
$plugins->add(new Plugin([
'id' => $instance->getIdentifier(),
'name' => $instance->getName(),
'identifier' => $instance->getIdentifier(),
'enabled' => $hasApiProvider && $apiProvider->enabled,
'file_path' => $file->getPathname(),
'has_api_provider' => $hasApiProvider,
'configured' => $hasApiProvider
]));
} catch (Exception $e) {
Log::error('Error loading plugin for ListPlugins page.', ['plugin_file' => $file->getPathname(), 'error' => $e->getMessage()]);
}
}
}
}
return $plugins;
}
public function table(Table $table): Table
{
return $table
->query(fn () => $this->getPlugins())
->columns([
TextColumn::make('name')
->label('Name')
->searchable()
->sortable(),
TextColumn::make('identifier')
->label('Identifier'),
IconColumn::make('enabled')
->label('Enabled')
->boolean(),
IconColumn::make('configured')
->label('Configured')
->boolean()
->tooltip(fn ($record) => $record->configured ? 'Has ApiProvider record' : 'No ApiProvider record'),
TextColumn::make('file_path')
->label('File Path')
->toggleable(isToggledHiddenByDefault: true),
])
->actions([
Action::make('toggle_enabled')
->label(fn($record) => $record->enabled ? 'Disable' : 'Enable')
->icon(fn($record) => $record->enabled ? 'heroicon-o-x-circle' :
'heroicon-o-check-circle')
->action(function ($record) {
// Action logic here
}),
Action::make('delete')
->label('Delete')
->icon('heroicon-o-trash')
->color('danger')
->requiresConfirmation()
->action(function ($record) {
// Action logic here
}),
]);
}
public function render()
{
return view('livewire.list-plugins');
}
public function currentlyValidatingForm(\Filament\Forms\ComponentContainer|null $form): void
{
// No form validation needed for this component
}
}

66
app/Models/Plugin.php Normal file
View File

@@ -0,0 +1,66 @@
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
use App\Api\Plugins\ApiPluginInterface;
use App\Models\ApiProvider;
use Illuminate\Support\Facades\File;
use Exception;
use ReflectionClass;
class Plugin extends Model
{
protected $table = null; // No actual table
protected $guarded = []; // Allow mass assignment for all attributes
public $incrementing = false;
protected $keyType = 'string';
protected $primaryKey = 'id';
public static function getAllPlugins()
{
$plugins = [];
$path = app_path('Api/Plugins');
if (File::exists($path)) {
$files = File::files($path);
foreach ($files as $file) {
$filename = $file->getFilenameWithoutExtension();
if (in_array($filename, ['ApiPluginInterface', 'LoggablePlugin', 'PluginLoader'])) {
continue;
}
$class = 'App\Api\Plugins\\' . $filename;
if (class_exists($class) && in_array(ApiPluginInterface::class, class_implements($class))) {
try {
// Check if there's an ApiProvider for this plugin
$apiProvider = ApiProvider::where('plugin', $filename)->first();
$hasApiProvider = $apiProvider !== null;
// Get plugin information without instantiating the class
// This avoids issues with plugins requiring ApiProvider in constructor
$reflection = new ReflectionClass($class);
$instance = $reflection->newInstanceWithoutConstructor();
$plugins[] = new self([
'id' => $instance->getIdentifier(),
'name' => $instance->getName(),
'identifier' => $instance->getIdentifier(),
'enabled' => $hasApiProvider && $apiProvider && $apiProvider->enabled,
'file_path' => $file->getPathname(),
'has_api_provider' => $hasApiProvider,
'configured' => $hasApiProvider
]);
} catch (Exception $e) {
// Log error or handle as needed
}
}
}
}
return new \Illuminate\Database\Eloquent\Collection($plugins);
}
}

View File

@@ -21,6 +21,7 @@ use App\Filament\Resources\StyleResource;
use App\Filament\Resources\SettingResource\Pages\Settings;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\HtmlString;
use App\Filament\Resources\PluginResource;
class AdminPanelProvider extends PanelProvider
{
@@ -40,13 +41,13 @@ class AdminPanelProvider extends PanelProvider
])
->discoverResources(in: app_path('Filament/Resources'), for: 'App\\Filament\\Resources')
->resources([
\App\Filament\Resources\ApiProviderResource::class,
// PluginResource::class, // Removed as it's a custom page now
])
->discoverPages(in: app_path('Filament/Pages'), for: 'App\\Filament\\Pages')
->pages([
Pages\Dashboard::class,
\App\Filament\Pages\InstallPluginPage::class,
// \App\Filament\Pages\ListPlugins::class, // Removed duplicate entry
])
->discoverWidgets(in: app_path('Filament/Widgets'), for: 'App\\Filament\\Widgets')
->widgets([
@@ -74,7 +75,7 @@ class AdminPanelProvider extends PanelProvider
->profile();
if (Auth::check()) {
$user = Auth->user();
$user = Auth::user();
if ($user->theme_preference === 'dark') {
$panel->darkMode();
} else {

View File

@@ -0,0 +1,44 @@
@extends('layouts.app')
@section('content')
<div class="container mx-auto px-4 py-8">
<h1 class="text-3xl font-bold mb-6">Plugin List</h1>
<div class="bg-white shadow-md rounded-lg overflow-hidden">
<table class="min-w-full divide-y divide-gray-200">
<thead class="bg-gray-50">
<tr>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Name</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Identifier</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Enabled</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Configured</th>
<th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">File Path</th>
</tr>
</thead>
<tbody class="bg-white divide-y divide-gray-200">
@foreach($plugins as $plugin)
<tr>
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900">{{ $plugin->name }}</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">{{ $plugin->identifier }}</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
@if($plugin->enabled)
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">Enabled</span>
@else
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-red-100 text-red-800">Disabled</span>
@endif
</td>
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
@if($plugin->configured)
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-green-100 text-green-800">Configured</span>
@else
<span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-yellow-100 text-yellow-800">Not Configured</span>
@endif
</td>
<td class="px-6 py-4 text-sm text-gray-500">{{ $plugin->file_path }}</td>
</tr>
@endforeach
</tbody>
</table>
</div>
</div>
@endsection

View File

@@ -1,3 +0,0 @@
<x-filament-panels::page>
{{ $this->table }}
</x-filament-panels::page>

View File

@@ -0,0 +1,64 @@
<x-filament-panels::page>
<x-filament::section>
<x-slot name="heading">
Plugins
</x-slot>
<x-filament-tables::table>
<thead>
<tr>
<x-filament-tables::header-cell>
Name
</x-filament-tables::header-cell>
<x-filament-tables::header-cell>
Identifier
</x-filament-tables::header-cell>
<x-filament-tables::header-cell>
Enabled
</x-filament-tables::header-cell>
<x-filament-tables::header-cell>
Configured
</x-filament-tables::header-cell>
</tr>
</thead>
<tbody>
@foreach ($this->getTableRecords() as $record)
<x-filament-tables::row>
<x-filament-tables::cell>
{{ $record->name }}
</x-filament-tables::cell>
<x-filament-tables::cell>
{{ $record->identifier }}
</x-filament-tables::cell>
<x-filament-tables::cell>
@if ($record->enabled)
<x-filament::icon
icon="heroicon-o-check-circle"
class="text-success-500"
/>
@else
<x-filament::icon
icon="heroicon-o-x-circle"
class="text-danger-500"
/>
@endif
</x-filament-tables::cell>
<x-filament-tables::cell>
@if ($record->configured)
<x-filament::icon
icon="heroicon-o-check-circle"
class="text-success-500"
/>
@else
<x-filament::icon
icon="heroicon-o-x-circle"
class="text-danger-500"
/>
@endif
</x-filament-tables::cell>
</x-filament-tables::row>
@endforeach
</tbody>
</x-filament-tables::table>
</x-filament::section>
</x-filament-panels::page>

View File

@@ -0,0 +1,5 @@
<div>
<x-filament-panels::page>
{{ $this->table }}
</x-filament-panels::page>
</div>

View File

@@ -5,6 +5,7 @@ use Illuminate\Foundation\Application;
use Illuminate\Support\Facades\Route;
use Inertia\Inertia;
use App\Http\Controllers\LocaleController;
use App\Http\Controllers\Admin\PluginController;
/*
|--------------------------------------------------------------------------
@@ -27,6 +28,9 @@ Route::middleware('auth')->group(function () {
Route::get('/profile', [ProfileController::class, 'edit'])->name('profile.edit');
Route::patch('/profile', [ProfileController::class, 'update'])->name('profile.update');
Route::delete('/profile', [ProfileController::class, 'destroy'])->name('profile.destroy');
// Plugin routes
Route::get('/admin/plugins', [PluginController::class, 'index'])->name('admin.plugins.index');
});
require __DIR__.'/auth.php';