feat(Production): Complete production deployment infrastructure

- Add comprehensive health check system with multiple endpoints
- Add Prometheus metrics endpoint
- Add production logging configurations (5 strategies)
- Add complete deployment documentation suite:
  * QUICKSTART.md - 30-minute deployment guide
  * DEPLOYMENT_CHECKLIST.md - Printable verification checklist
  * DEPLOYMENT_WORKFLOW.md - Complete deployment lifecycle
  * PRODUCTION_DEPLOYMENT.md - Comprehensive technical reference
  * production-logging.md - Logging configuration guide
  * ANSIBLE_DEPLOYMENT.md - Infrastructure as Code automation
  * README.md - Navigation hub
  * DEPLOYMENT_SUMMARY.md - Executive summary
- Add deployment scripts and automation
- Add DEPLOYMENT_PLAN.md - Concrete plan for immediate deployment
- Update README with production-ready features

All production infrastructure is now complete and ready for deployment.
This commit is contained in:
2025-10-25 19:18:37 +02:00
parent caa85db796
commit fc3d7e6357
83016 changed files with 378904 additions and 20919 deletions

View File

@@ -47,7 +47,7 @@ final readonly class AdminApiHandler
// Convert to arrays
$items = array_map(
fn($item) => method_exists($item, 'toArray') ? $item->toArray() : (array) $item,
fn ($item) => method_exists($item, 'toArray') ? $item->toArray() : (array) $item,
$paginatedData
);
@@ -147,7 +147,7 @@ final readonly class AdminApiHandler
try {
$deleted = $repository->delete($id);
if (!$deleted) {
if (! $deleted) {
return new JsonResult([
'success' => false,
'error' => 'Resource not found',

View File

@@ -4,8 +4,8 @@ declare(strict_types=1);
namespace App\Framework\Admin\Factories;
use App\Framework\Admin\ValueObjects\AdminFormConfig;
use App\Framework\Admin\FormFields\FormFieldFactory;
use App\Framework\Admin\ValueObjects\AdminFormConfig;
use App\Framework\Http\Session\FormIdGenerator;
use App\Framework\View\FormBuilder;
@@ -19,7 +19,8 @@ final readonly class AdminFormFactory
public function __construct(
private FormIdGenerator $formIdGenerator,
private FormFieldFactory $fieldFactory
) {}
) {
}
public function create(AdminFormConfig $config): FormBuilder
{
@@ -36,7 +37,7 @@ final readonly class AdminFormFactory
$fieldConfigWithValue = [
...$fieldConfig,
'name' => $name,
'value' => $config->data[$name] ?? null
'value' => $config->data[$name] ?? null,
];
// Create field using factory

View File

@@ -5,9 +5,9 @@ declare(strict_types=1);
namespace App\Framework\Admin\Factories;
use App\Framework\Admin\ValueObjects\AdminTableConfig;
use App\Framework\View\Table\Formatters;
use App\Framework\View\Table\Table;
use App\Framework\View\Table\TableOptions;
use App\Framework\View\Table\Formatters;
/**
* Admin Table Factory

View File

@@ -4,10 +4,10 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields\Fields;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\Admin\FormFields\FormField;
use App\Framework\Admin\FormFields\ValueObjects\FieldAttributes;
use App\Framework\Admin\FormFields\ValueObjects\FieldMetadata;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\View\FormBuilder;
use App\Framework\View\ValueObjects\FormElement;
@@ -24,7 +24,8 @@ final readonly class CheckboxField implements FormField
private FieldWrapper $wrapper,
private mixed $value = null,
private string $checkedValue = '1'
) {}
) {
}
public static function create(
string $name,

View File

@@ -4,10 +4,10 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields\Fields;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\Admin\FormFields\FormField;
use App\Framework\Admin\FormFields\ValueObjects\FieldAttributes;
use App\Framework\Admin\FormFields\ValueObjects\FieldMetadata;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\View\FormBuilder;
use App\Framework\View\ValueObjects\FormElement;
use DateTimeInterface;
@@ -25,7 +25,8 @@ final readonly class DateTimeField implements FormField
private FieldWrapper $wrapper,
private mixed $value = null,
private string $type = 'datetime-local'
) {}
) {
}
public static function create(
string $name,

View File

@@ -4,10 +4,10 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields\Fields;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\Admin\FormFields\FormField;
use App\Framework\Admin\FormFields\ValueObjects\FieldAttributes;
use App\Framework\Admin\FormFields\ValueObjects\FieldMetadata;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\View\FormBuilder;
use App\Framework\View\ValueObjects\FormElement;
@@ -23,7 +23,8 @@ final readonly class EmailField implements FormField
private FieldAttributes $attributes,
private FieldWrapper $wrapper,
private mixed $value = null
) {}
) {
}
public static function create(
string $name,

View File

@@ -21,7 +21,8 @@ final readonly class HiddenField implements FormField
private FieldMetadata $metadata,
private FieldAttributes $attributes,
private mixed $value = null
) {}
) {
}
public static function create(
string $name,

View File

@@ -4,10 +4,10 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields\Fields;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\Admin\FormFields\FormField;
use App\Framework\Admin\FormFields\ValueObjects\FieldAttributes;
use App\Framework\Admin\FormFields\ValueObjects\FieldMetadata;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\View\FormBuilder;
use App\Framework\View\ValueObjects\FormElement;
@@ -26,7 +26,8 @@ final readonly class NumberField implements FormField
private ?int $min = null,
private ?int $max = null,
private ?int $step = null
) {}
) {
}
public static function create(
string $name,

View File

@@ -4,11 +4,11 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields\Fields;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\Admin\FormFields\FormField;
use App\Framework\Admin\FormFields\ValueObjects\FieldAttributes;
use App\Framework\Admin\FormFields\ValueObjects\FieldMetadata;
use App\Framework\Admin\FormFields\ValueObjects\FieldOptions;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\View\FormBuilder;
use App\Framework\View\ValueObjects\FormElement;
@@ -25,7 +25,8 @@ final readonly class SelectField implements FormField
private FieldWrapper $wrapper,
private FieldOptions $options,
private mixed $value = null
) {}
) {
}
public static function create(
string $name,

View File

@@ -4,10 +4,10 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields\Fields;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\Admin\FormFields\FormField;
use App\Framework\Admin\FormFields\ValueObjects\FieldAttributes;
use App\Framework\Admin\FormFields\ValueObjects\FieldMetadata;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\View\FormBuilder;
use App\Framework\View\ValueObjects\FormElement;
@@ -23,7 +23,8 @@ final readonly class TextField implements FormField
private FieldAttributes $attributes,
private FieldWrapper $wrapper,
private mixed $value = null
) {}
) {
}
public static function create(
string $name,

View File

@@ -4,10 +4,10 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields\Fields;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\Admin\FormFields\FormField;
use App\Framework\Admin\FormFields\ValueObjects\FieldAttributes;
use App\Framework\Admin\FormFields\ValueObjects\FieldMetadata;
use App\Framework\Admin\FormFields\Components\FieldWrapper;
use App\Framework\View\FormBuilder;
use App\Framework\View\ValueObjects\FormElement;
@@ -24,7 +24,8 @@ final readonly class TextareaField implements FormField
private FieldWrapper $wrapper,
private mixed $value = null,
private int $rows = 5
) {}
) {
}
public static function create(
string $name,

View File

@@ -4,16 +4,16 @@ declare(strict_types=1);
namespace App\Framework\Admin\FormFields;
use App\Framework\Admin\FormFields\Fields\TextField;
use App\Framework\Admin\FormFields\Fields\EmailField;
use App\Framework\Admin\FormFields\Fields\NumberField;
use App\Framework\Admin\FormFields\Fields\TextareaField;
use App\Framework\Admin\FormFields\Fields\SelectField;
use App\Framework\Admin\FormFields\Fields\DateTimeField;
use App\Framework\Admin\FormFields\Fields\CheckboxField;
use App\Framework\Admin\FormFields\Fields\DateTimeField;
use App\Framework\Admin\FormFields\Fields\EmailField;
use App\Framework\Admin\FormFields\Fields\HiddenField;
use App\Framework\Exception\FrameworkException;
use App\Framework\Admin\FormFields\Fields\NumberField;
use App\Framework\Admin\FormFields\Fields\SelectField;
use App\Framework\Admin\FormFields\Fields\TextareaField;
use App\Framework\Admin\FormFields\Fields\TextField;
use App\Framework\Exception\ErrorCode;
use App\Framework\Exception\FrameworkException;
/**
* Form Field Factory
@@ -140,7 +140,7 @@ final readonly class FormFieldFactory
public function createMultiple(array $fieldsConfig): array
{
return array_map(
fn(array $config) => $this->createFromConfig($config),
fn (array $config) => $this->createFromConfig($config),
$fieldsConfig
);
}

View File

@@ -21,7 +21,8 @@ final readonly class FieldAttributes
public bool $required = false,
public ?string $placeholder = null,
public array $additional = []
) {}
) {
}
/**
* Convert attributes to array for rendering
@@ -34,7 +35,7 @@ final readonly class FieldAttributes
'name' => $this->name,
'id' => $this->id,
'class' => $this->class,
...$this->additional
...$this->additional,
];
if ($this->required) {

View File

@@ -15,5 +15,6 @@ final readonly class FieldMetadata
public string $name,
public string $label,
public ?string $help = null
) {}
) {
}
}

View File

@@ -18,7 +18,8 @@ final readonly class FieldOptions
public function __construct(
public array $options,
public ?string $placeholder = null
) {}
) {
}
/**
* Convert options to HTML option elements

View File

@@ -23,7 +23,8 @@ final readonly class CrudService
public function __construct(
private TemplateRenderer $renderer,
private AdminFormFactory $formFactory
) {}
) {
}
/**
* Render index view

View File

@@ -36,7 +36,8 @@ final readonly class CrudConfig
public ?array $filters = null,
public ?array $bulkActions = null,
public bool $searchable = true,
) {}
) {
}
public static function forResource(
string $resource,