Skip to content

Auto-Discovery

The Pollora Discovery system provides automatic component discovery in your application, WordPress themes, and Laravel modules. It uses a modern architecture inspired by Tempest Framework while leveraging Spatie’s structure discovery capabilities to identify and process different types of components according to your needs.

The Discovery system automatically scans your codebase to find and register components like:

  • Custom Post Types with #[PostType] attributes
  • Custom Taxonomies with #[Taxonomy] attributes
  • Scheduled Tasks with #[Schedule] attributes
  • WordPress Hooks with #[Action] and #[Filter] attributes
  • REST API Routes with #[WpRestRoute] attributes
  • Custom Components through extensible discovery classes
  • Modern API: Inspired by Tempest Framework’s discovery system
  • Performance Optimized: Built on Spatie’s structure discoverer with caching
  • Domain-Driven Design: Clean architecture with proper separation of concerns
  • Extensible: Easy to create custom discovery classes for any component type
  • Type Safe: Full PHPDoc coverage and strict type declarations
  • Laravel Integration: Seamless integration with Laravel’s service container
┌─────────────────────────────────────────────────────────────┐
│ DiscoveryManager │
│ (High-level API) │
└──────────────────────────┬──────────────────────────────────┘
┌──────────────────────────▼──────────────────────────────────┐
│ DiscoveryEngine │
│ (Orchestration Layer) │
└──────────────────────────┬──────────────────────────────────┘
┌──────────────────────┼──────────────────────┐
│ │ │
┌───▼─────────┐ ┌─────────▼──────┐ ┌───────────▼──────────┐
│ Discovery │ │ Discovery │ │ Discovery Cache │
│ Locations │ │ Classes │ │ (Laravel Cache) │
└─────────────┘ └────────────────┘ └──────────────────────┘
  1. Registration: Discovery classes are registered with unique identifiers
  2. Location Setup: Discovery locations (namespaces + paths) are configured
  3. Discovery Phase: Engine scans all locations using registered discoveries
  4. Application Phase: Discovered items are applied/registered with the framework
  5. Caching: Results are cached for performance in production

Location: Pollora\PostType\Infrastructure\Services\PostTypeDiscovery Discovers: Classes with #[PostType] attributes

#[PostType('product', ['public' => true])]
class Product extends AbstractPostType
{
// Post type implementation
}

Location: Pollora\Taxonomy\Infrastructure\Services\TaxonomyDiscovery Discovers: Classes with #[Taxonomy] attributes

#[Taxonomy('product-category', ['hierarchical' => true])]
class ProductCategory extends AbstractTaxonomy
{
// Taxonomy implementation
}

Location: Pollora\Schedule\Infrastructure\Services\ScheduleDiscovery Discovers: Methods with #[Schedule] attributes

class TaskManager
{
#[Schedule('daily')]
public function cleanupTempFiles(): void
{
// Scheduled task implementation
}
}

Location: Pollora\Hook\Infrastructure\Services\HookDiscovery Discovers: Methods with #[Action] or #[Filter] attributes

class UserHooks
{
#[Action('user_register')]
public function onUserRegister(int $userId): void
{
// Action handler
}
#[Filter('the_content')]
public function filterContent(string $content): string
{
// Filter handler
return $content;
}
}

Location: Pollora\WpRest\Infrastructure\Services\WpRestDiscovery Discovers: Methods with #[WpRestRoute] attributes

class ApiController
{
#[WpRestRoute('v1', '/products', ['GET'])]
public function getProducts(): array
{
// REST endpoint implementation
return [];
}
}
use Pollora\Discovery\Application\Services\DiscoveryManager;
// Get the discovery manager from container
$discoveryManager = app(DiscoveryManager::class);
// Add discovery locations
$discoveryManager->addLocation('MyTheme\\', get_stylesheet_directory() . '/app');
// Run discovery and apply all results
$discoveryManager->run();
// Or run discovery phases separately
$discoveryManager->discover(); // Discovery phase only
$discoveryManager->apply(); // Application phase only
// Check if a discovery is registered
if ($discoveryManager->hasDiscovery('post_types')) {
$items = $discoveryManager->getDiscoveredItems('post_types');
echo "Found " . count($items) . " post types\n";
}
// Get all registered discoveries
$discoveries = $discoveryManager->getDiscoveries();
// Get all discovery locations
$locations = $discoveryManager->getLocations();
// Clear discovery caches
$discoveryManager->clearCache();
<?php
declare(strict_types=1);
namespace MyTheme\Discovery;
use MyTheme\Attributes\CustomComponent;
use Pollora\Discovery\Domain\Contracts\DiscoveryInterface;
use Pollora\Discovery\Domain\Contracts\DiscoveryLocationInterface;
use Pollora\Discovery\Domain\Services\IsDiscovery;
use Spatie\StructureDiscoverer\Data\DiscoveredStructure;
/**
* Custom Component Discovery
*
* Discovers classes with the CustomComponent attribute
*/
final class CustomComponentDiscovery implements DiscoveryInterface
{
use IsDiscovery;
public function __construct(
private readonly MyComponentService $componentService
) {}
public function discover(DiscoveryLocationInterface $location, DiscoveredStructure $structure): void
{
// Only process classes
if (!$structure instanceof \Spatie\StructureDiscoverer\Data\DiscoveredClass) {
return;
}
// Check for our custom attribute
$customAttribute = null;
foreach ($structure->attributes as $attribute) {
if ($attribute->name === CustomComponent::class) {
$customAttribute = $attribute;
break;
}
}
if ($customAttribute === null || $structure->isAbstract) {
return;
}
// Collect the discovered class
$this->getItems()->add($location, [
'class' => $structure->name,
'attribute' => $customAttribute,
'structure' => $structure,
]);
}
public function apply(): void
{
foreach ($this->getItems() as $discoveredItem) {
[
'class' => $className,
'attribute' => $attribute,
'structure' => $structure
] = $discoveredItem;
try {
// Register the component through your service
$this->componentService->registerComponent($className);
} catch (\Throwable $e) {
error_log("Failed to register component {$className}: " . $e->getMessage());
}
}
}
public function getIdentifier(): string
{
return 'custom_components';
}
}
<?php
declare(strict_types=1);
namespace MyTheme\Discovery;
use Pollora\Discovery\Domain\Contracts\DiscoversPathInterface;
use Pollora\Discovery\Domain\Contracts\DiscoveryLocationInterface;
use Pollora\Discovery\Domain\Services\IsDiscovery;
use Spatie\StructureDiscoverer\Data\DiscoveredStructure;
/**
* Template Discovery
*
* Discovers both PHP classes and template files
*/
final class TemplateDiscovery implements DiscoversPathInterface
{
use IsDiscovery;
public function discover(DiscoveryLocationInterface $location, DiscoveredStructure $structure): void
{
// Discover PHP template classes
if ($structure instanceof \Spatie\StructureDiscoverer\Data\DiscoveredClass) {
if (str_contains($structure->name, 'Template')) {
$this->getItems()->add($location, [
'type' => 'class',
'class' => $structure->name,
]);
}
}
}
public function discoverPath(DiscoveryLocationInterface $location, string $path): void
{
// Discover template files
if (str_ends_with($path, '.blade.php') || str_ends_with($path, '.php')) {
if (str_contains($path, '/templates/')) {
$this->getItems()->add($location, [
'type' => 'file',
'path' => $path,
]);
}
}
}
public function apply(): void
{
foreach ($this->getItems() as $discoveredItem) {
$type = $discoveredItem['type'];
if ($type === 'class') {
// Register PHP template class
$this->registerTemplateClass($discoveredItem['class']);
} elseif ($type === 'file') {
// Register template file
$this->registerTemplateFile($discoveredItem['path']);
}
}
}
public function getIdentifier(): string
{
return 'templates';
}
}
<?php
namespace MyTheme\Providers;
use Illuminate\Support\ServiceProvider;
use MyTheme\Discovery\CustomComponentDiscovery;
use Pollora\Discovery\Domain\Contracts\DiscoveryEngineInterface;
class ThemeServiceProvider extends ServiceProvider
{
public function register(): void
{
// Register your discovery class
$this->app->singleton(CustomComponentDiscovery::class);
}
public function boot(): void
{
/** @var DiscoveryEngineInterface $engine */
$engine = $this->app->make(DiscoveryEngineInterface::class);
// Add your discovery to the engine
$engine->addDiscovery('custom_components', $this->app->make(CustomComponentDiscovery::class));
// Add theme locations for discovery
$engine->addLocation(
new \Pollora\Discovery\Domain\Models\DiscoveryLocation(
'MyTheme\\',
get_stylesheet_directory() . '/app'
)
);
}
}
use Pollora\Discovery\Domain\Contracts\DiscoveryEngineInterface;
use Pollora\Discovery\Domain\Models\DiscoveryLocation;
/** @var DiscoveryEngineInterface $engine */
$engine = app(DiscoveryEngineInterface::class);
// Add discovery locations
$engine->addLocation(new DiscoveryLocation('App\\', app_path()));
$engine->addLocation(new DiscoveryLocation('MyTheme\\', get_stylesheet_directory() . '/app'));
// Add custom discoveries
$engine->addDiscovery('my_discovery', MyDiscovery::class);
// Configure caching
$cache = app(\Pollora\Discovery\Domain\Contracts\DiscoveryCacheInterface::class);
$engine->withCache($cache);
// Run discovery
$engine->run(); // Discovery + Apply
// or
$engine->discover(); // Discovery only
$engine->apply(); // Apply only
// Get specific discovery
$postTypeDiscovery = $engine->getDiscovery('post_types');
$discoveredItems = $postTypeDiscovery->getItems()->all();
// Get all discoveries
$allDiscoveries = $engine->getDiscoveries();
// Get all locations
$locations = $engine->getLocations();

The system automatically caches discovery results using Laravel’s cache system:

use Pollora\Discovery\Infrastructure\Adapters\LaravelDiscoveryCache;
// Cache is automatically configured in the service provider
// but you can customize it:
$cache = new LaravelDiscoveryCache(
cache: app('cache.store'),
prefix: 'my_app.discovery.',
defaultTtl: 3600 // 1 hour
);
use Pollora\Discovery\Domain\Contracts\DiscoveryCacheInterface;
/** @var DiscoveryCacheInterface $cache */
$cache = app(DiscoveryCacheInterface::class);
// Check if cached
$cacheKey = $cache->generateKey('post_types', $locations);
if ($cache->has($cacheKey)) {
$items = $cache->get($cacheKey);
}
// Clear specific cache
$cache->forget($cacheKey);
// Clear all discovery caches
$cache->flush();
Terminal window
# Run all discoveries
php artisan discovery:run
# Run specific discovery
php artisan discovery:run --discovery=post_types
# Clear cache before running
php artisan discovery:run --clear-cache
# Run with verbose output
php artisan discovery:run -v
Terminal window
# Clear all discovery caches
php artisan discovery:clear
// In your .env file
DISCOVERY_CACHE_TTL=3600 # Cache time-to-live in seconds
DISCOVERY_CACHE_PREFIX=app.discovery. # Cache key prefix
// In a service provider
public function boot(): void
{
/** @var DiscoveryManager $manager */
$manager = $this->app->make(DiscoveryManager::class);
// Add application-specific locations
$manager->addLocations([
['namespace' => 'App\\', 'path' => app_path()],
['namespace' => 'MyTheme\\', 'path' => get_stylesheet_directory() . '/app'],
]);
// Add custom discoveries
$manager->addDiscoveries([
'my_components' => MyComponentDiscovery::class,
'my_services' => MyServiceDiscovery::class,
]);
// Run discovery on application boot
$manager->run();
}
<?php
namespace MyTheme\Discovery;
use MyTheme\Attributes\Component;
use Pollora\Discovery\Domain\Contracts\DiscoveryInterface;
use Pollora\Discovery\Domain\Contracts\DiscoveryLocationInterface;
use Pollora\Discovery\Domain\Services\IsDiscovery;
use Spatie\StructureDiscoverer\Data\DiscoveredStructure;
final class ComponentDiscovery implements DiscoveryInterface
{
use IsDiscovery;
public function discover(DiscoveryLocationInterface $location, DiscoveredStructure $structure): void
{
if (!$structure instanceof \Spatie\StructureDiscoverer\Data\DiscoveredClass) {
return;
}
foreach ($structure->attributes as $attribute) {
if ($attribute->name === Component::class) {
$this->getItems()->add($location, [
'class' => $structure->name,
'attribute' => $attribute,
]);
break;
}
}
}
public function apply(): void
{
foreach ($this->getItems() as $item) {
$this->registerComponent($item['class'], $item['attribute']);
}
}
public function getIdentifier(): string
{
return 'theme_components';
}
private function registerComponent(string $className, $attribute): void
{
// Component registration logic
add_action('wp_enqueue_scripts', function () use ($className) {
$component = app($className);
$component->register();
});
}
}
<?php
namespace MyPlugin\Discovery;
use MyPlugin\Contracts\PluginService;
use Pollora\Discovery\Domain\Contracts\DiscoveryInterface;
use Pollora\Discovery\Domain\Contracts\DiscoveryLocationInterface;
use Pollora\Discovery\Domain\Services\IsDiscovery;
use Spatie\StructureDiscoverer\Data\DiscoveredStructure;
final class ServiceDiscovery implements DiscoveryInterface
{
use IsDiscovery;
public function discover(DiscoveryLocationInterface $location, DiscoveredStructure $structure): void
{
if (!$structure instanceof \Spatie\StructureDiscoverer\Data\DiscoveredClass) {
return;
}
// Check if class implements PluginService
if (in_array(PluginService::class, $structure->implements)) {
$this->getItems()->add($location, ['class' => $structure->name]);
}
}
public function apply(): void
{
foreach ($this->getItems() as $item) {
$serviceClass = $item['class'];
// Register as singleton in container
app()->singleton($serviceClass);
// Auto-bind to interface if it exists
$interfaces = class_implements($serviceClass);
foreach ($interfaces as $interface) {
if ($interface !== PluginService::class) {
app()->bind($interface, $serviceClass);
}
}
}
}
public function getIdentifier(): string
{
return 'plugin_services';
}
}
  1. Namespace: Pollora\DiscovererPollora\Discovery
  2. API: PolloraDiscover::scout()DiscoveryManager::run()
  3. Architecture: Scout-based → Discovery-based
  4. Placement: Central package → Module-specific discoveries
  1. Update Service Provider Registration:
// Old
$this->app->register(DiscovererServiceProvider::class);
// New
$this->app->register(DiscoveryServiceProvider::class);
  1. Convert Scout Classes to Discovery Classes:
// Old Scout
class MyScout extends AbstractPolloraScout
{
protected function criteria(Discover $discover): Discover
{
return $discover->classes()->implementing(MyInterface::class);
}
}
// New Discovery
class MyDiscovery implements DiscoveryInterface
{
use IsDiscovery;
public function discover(DiscoveryLocationInterface $location, DiscoveredStructure $structure): void
{
if ($structure instanceof DiscoveredClass) {
if (in_array(MyInterface::class, $structure->implements)) {
$this->getItems()->add($location, ['class' => $structure->name]);
}
}
}
public function apply(): void
{
foreach ($this->getItems() as $item) {
// Register discovered class
}
}
public function getIdentifier(): string
{
return 'my_discovery';
}
}
  1. Update Usage:
// Old
PolloraDiscover::register('my_scout', MyScout::class);
$classes = PolloraDiscover::scout('my_scout');
// New
$manager = app(DiscoveryManager::class);
$manager->addDiscovery('my_discovery', MyDiscovery::class);
$manager->run();
$classes = $manager->getDiscoveredItems('my_discovery');
DiscoveryNotFoundException: Discovery not found with identifier: my_discovery

Solution: Ensure the discovery is registered before use:

/** @var DiscoveryManager $manager */
$manager = app(DiscoveryManager::class);
if (!$manager->hasDiscovery('my_discovery')) {
$manager->addDiscovery('my_discovery', MyDiscovery::class);
}

Possible causes:

  • Discovery locations not configured
  • Discovery class not properly implementing interface
  • Attribute/interface not found

Debug:

// Enable verbose console output
php artisan discovery:run -v
// Check discovery locations
$manager = app(DiscoveryManager::class);
$locations = $manager->getLocations();
dump($locations);
// Check discovered items
$items = $manager->getDiscoveredItems('my_discovery');
dump($items);

Solution: Clear discovery cache:

Terminal window
# Via console
php artisan discovery:clear
# Via code
app(DiscoveryManager::class)->clearCache();

Solution: Optimize discovery locations:

// Be specific about discovery paths
$manager->addLocation('App\\Services\\', app_path('Services'));
$manager->addLocation('App\\Models\\', app_path('Models'));
// Instead of scanning entire app directory
// $manager->addLocation('App\\', app_path());

Enable debug logging during development:

try {
$manager = app(DiscoveryManager::class);
$manager->run();
$discoveries = $manager->getDiscoveries();
foreach ($discoveries as $identifier => $discovery) {
$count = count($manager->getDiscoveredItems($identifier));
error_log("Discovery '{$identifier}' found {$count} items");
}
} catch (\Throwable $e) {
error_log('Discovery error: ' . $e->getMessage());
error_log('Stack trace: ' . $e->getTraceAsString());
}
$start = microtime(true);
$manager = app(DiscoveryManager::class);
$manager->run();
$duration = (microtime(true) - $start) * 1000;
error_log("Discovery completed in {$duration}ms");