• Add Service Providers loading from config/providers.php and merge with runtime config; ensure AppServiceProvider boots and contributes routes • Create RouteGroups and guard module route includes in routes/web.php; update Kernel to auto-mount module routes and apply provider routes • Implement create:module as a console Command (extends Phred\Console\Command): ◦ Args: name, prefix; Flags: --update-composer, --no-dump ◦ Stable root resolution (dirname(DIR, 2)); robust args/flags handling under ArrayInput ◦ Scaffolds module dirs (Controllers, Views, Templates, Routes, Providers, etc.), ensures Controllers exists, adds .gitkeep ◦ Writes Provider, View, Controller, Template stubs (fix variable interpolation via placeholders) ◦ Appends guarded include snippet to routes/web.php ◦ Optional composer PSR-4 mapping update (+ backup) and optional autoload dump ◦ Prevents providers.php corruption via name validation and existence checks • Add URL extension negotiation middleware tweaks: ◦ Only set Accept for .json (and future .xml), never for none/php ◦ Never override existing Accept header • Add MVC base classes (Controller, APIController, ViewController, View, ViewWithDefaultTemplate); update ViewController signature and View render contract • Add tests: ◦ CreateModuleCommandTest with setup/teardown to snapshot/restore routes/web.php and composer.json; asserts scaffold and PSR-4 mapping ◦ ProviderRouteTest for provider-contributed route ◦ UrlExtensionNegotiationTest sets API_FORMAT=rest and asserts content-type behavior ◦ MvcViewTest validates transformData+render • Fix config/providers.php syntax and add comment placeholder for modules • Update README: M5/M6/M7 docs, MVC examples, template selection conventions, modules section, URL extension negotiation, and module creation workflow • Update MILESTONES.md: mark M6/M7 complete; add M8 task for register:orm; note M12 XML extension support
104 lines
3.7 KiB
PHP
104 lines
3.7 KiB
PHP
<?php
|
|
declare(strict_types=1);
|
|
|
|
namespace Phred\Support;
|
|
|
|
use DI\Container;
|
|
use DI\ContainerBuilder;
|
|
use Phred\Support\Contracts\ConfigInterface;
|
|
use Phred\Support\Contracts\ServiceProviderInterface;
|
|
|
|
/**
|
|
* Loads and executes service providers in deterministic order.
|
|
* Order: core → app → modules
|
|
*/
|
|
final class ProviderRepository
|
|
{
|
|
/** @var list<ServiceProviderInterface> */
|
|
private array $providers = [];
|
|
|
|
public function __construct(private readonly ConfigInterface $config)
|
|
{
|
|
}
|
|
|
|
public function load(): void
|
|
{
|
|
$this->providers = [];
|
|
// Merge providers from config/providers.php file (authoritative) with any runtime Config entries
|
|
$fileCore = $fileApp = $fileModules = [];
|
|
$configFile = dirname(__DIR__, 2) . '/config/providers.php';
|
|
if (is_file($configFile)) {
|
|
/** @noinspection PhpIncludeInspection */
|
|
$arr = require $configFile;
|
|
if (is_array($arr)) {
|
|
$fileCore = (array)($arr['core'] ?? []);
|
|
$fileApp = (array)($arr['app'] ?? []);
|
|
$fileModules = (array)($arr['modules'] ?? []);
|
|
}
|
|
}
|
|
$core = array_values(array_unique(array_merge($fileCore, (array) Config::get('providers.core', []))));
|
|
$app = array_values(array_unique(array_merge($fileApp, (array) Config::get('providers.app', []))));
|
|
$modules = array_values(array_unique(array_merge($fileModules, (array) Config::get('providers.modules', []))));
|
|
|
|
foreach ([$core, $app, $modules] as $group) {
|
|
foreach ($group as $class) {
|
|
if (is_string($class) && class_exists($class)) {
|
|
$instance = new $class();
|
|
if ($instance instanceof ServiceProviderInterface) {
|
|
$this->providers[] = $instance;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Initial module discovery: scan modules/*/Providers/*ServiceProvider.php
|
|
$root = dirname(__DIR__, 2);
|
|
$modulesDir = $root . '/modules';
|
|
if (is_dir($modulesDir)) {
|
|
foreach (scandir($modulesDir) ?: [] as $entry) {
|
|
if ($entry === '.' || $entry === '..') {
|
|
continue;
|
|
}
|
|
$modulePath = $modulesDir . '/' . $entry;
|
|
if (!is_dir($modulePath)) {
|
|
continue;
|
|
}
|
|
$providersPath = $modulePath . '/Providers';
|
|
if (!is_dir($providersPath)) {
|
|
continue;
|
|
}
|
|
foreach (scandir($providersPath) ?: [] as $file) {
|
|
if ($file === '.' || $file === '..' || !str_ends_with($file, '.php')) {
|
|
continue;
|
|
}
|
|
$classBase = substr($file, 0, -4);
|
|
if (!str_ends_with($classBase, 'ServiceProvider')) {
|
|
continue;
|
|
}
|
|
$fqcn = "Project\\\\Modules\\\\{$entry}\\\\Providers\\\\{$classBase}";
|
|
if (class_exists($fqcn)) {
|
|
$instance = new $fqcn();
|
|
if ($instance instanceof ServiceProviderInterface) {
|
|
$this->providers[] = $instance;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public function registerAll(ContainerBuilder $builder): void
|
|
{
|
|
foreach ($this->providers as $provider) {
|
|
$provider->register($builder, $this->config);
|
|
}
|
|
}
|
|
|
|
public function bootAll(Container $container): void
|
|
{
|
|
foreach ($this->providers as $provider) {
|
|
$provider->boot($container);
|
|
}
|
|
}
|
|
}
|