bear / projection
Read-optimized query library for BEAR.Resource #[Embed]
Installs: 0
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 0
Forks: 0
Open Issues: 0
pkg:composer/bear/projection
Requires
- php: ^8.2
- ext-mysqli: *
- bear/resource: ^1.0
- ray/di: ^2.0
Requires (Dev)
- bamarni/composer-bin-plugin: ^1.8
- phpunit/phpunit: ^10.5
This package is auto-updated.
Last update: 2026-01-25 10:40:08 UTC
README
Read-optimized query library for BEAR.Resource #[Embed]
Overview
BEAR.Projection implements the BDR Pattern for the Read side, providing query:// scheme resources designed for #[Embed]. A companion to Ray.MediaQuery.
Features
- Embed-first design - Optimized for
#[Embed]in ResourceObject - Parallel SQL execution - Multiple
#[Embed]queries run concurrently - DI-powered Factory - Inject dependencies for data transformation
Requirements
- PHP 8.2+
- MySQL (uses mysqli for async query execution, not PDO)
Installation
composer require bear/projection
Usage
1. Define a Projection
namespace MyApp\Projection; final class UserProfile { public function __construct( public readonly string $id, public readonly string $name, public readonly int $age, public readonly string $avatarUrl, ) { } }
2. Create a Factory with DI
Factory can inject dependencies for data transformation and validation:
namespace MyApp\Projection; final class UserProfileFactory { public function __construct( private readonly AgeCalculator $ageCalculator, private readonly ImageUrlResolver $imageResolver, ) { } public function __invoke( string $id, string $name, string $birthDate, string $avatarPath, ): UserProfile { return new UserProfile( id: $id, name: $name, age: $this->ageCalculator->fromBirthDate($birthDate), avatarUrl: $this->imageResolver->resolve($avatarPath), ); } }
3. Write the SQL Query
-- var/sql/query/user_profile.sql SELECT id, name, birth_date, avatar_path FROM users WHERE id = :id
4. Configure the Module
use BEAR\Projection\Module\ProjectionModule; class AppModule extends AbstractModule { protected function configure(): void { $this->install(new ProjectionModule( sqlDir: __DIR__ . '/../var/sql/query', projectionNamespace: 'MyApp\\Projection\\', )); } }
5. Embed in Resource
use BEAR\Resource\Annotation\Embed; use BEAR\Resource\ResourceObject; class User extends ResourceObject { #[Embed(rel: 'profile', src: 'query://self/user_profile{?id}')] #[Embed(rel: 'orders', src: 'query://self/user_orders{?id}')] public function onGet(string $id): static { // Both queries execute in parallel return $this; } }
File Structure
var/sql/query/
user_profile.sql
user_orders.sql
src/Projection/
UserProfile.php
UserProfileFactory.php
UserOrders.php
UserOrdersFactory.php
Naming Conventions
| SQL file | Factory class | URI |
|---|---|---|
user_profile.sql |
UserProfileFactory |
query://self/user_profile |
user_orders.sql |
UserOrdersFactory |
query://self/user_orders |