maatify / shared-common
Fundamental contracts and shared abstractions for Maatify modules, including clock and telemetry interfaces.
Requires
- php: ^8.2
Requires (Dev)
- friendsofphp/php-cs-fixer: ^3.94
- php-di/php-di: ^7.0
- phpstan/phpstan: ^1.12
- phpunit/phpunit: ^11.0
- slim/psr7: ^1.8
README
Overview
The Maatify\SharedCommon module contains foundational contracts and abstractions that are intended to be shared across all Maatify modules (such as AdminKernel, Verification, etc.). Its primary goal is to provide unified interfaces for common cross-cutting concerns like time management, security contexts, permission mapping definitions, and telemetry, enabling consistent behavior and testing across the entire system.
Purpose
By depending on SharedCommon rather than framework-specific implementations or raw PHP functions (like time() or date()), other modules can remain truly framework-agnostic. This module defines the "how we communicate" contracts for basic system realities.
Module Structure
Modules/SharedCommon/ ├── Bootstrap/ # Dependency Injection bindings │ └── SharedCommonBindings.php ├── Contracts/ # Core interfaces for time, telemetry, security, and shared module extensions │ ├── ClockInterface.php │ ├── Security/ # Framework-neutral security extension contracts │ │ ├── PermissionMapProviderInterface.php │ │ ├── PermissionRequirementDefinition.php │ │ └── ProvidesPermissionMapsInterface.php │ ├── SecurityEventContextInterface.php │ └── TelemetryContextInterface.php ├── Infrastructure/ # Default implementations of contracts │ └── SystemClock.php ├── Path/ # Common application path resolution utilities │ └── AppPaths.php ├── docs/ # Architectural and integration documentation └── composer.json # Standalone package metadata
Quick Usage
To quickly integrate the default implementations of this module into your dependency injection container:
use Maatify\SharedCommon\Bootstrap\SharedCommonBindings; use DI\ContainerBuilder; use Maatify\SharedCommon\Contracts\ClockInterface; $builder = new ContainerBuilder(); // Register the bindings for SharedCommon contracts SharedCommonBindings::register($builder); $container = $builder->build(); // Resolve the Clock /** @var ClockInterface $clock */ $clock = $container->get(ClockInterface::class); // Get the current time as a DateTimeImmutable object $now = $clock->now(); echo $now->format('Y-m-d H:i:s');
Permission Mapping Contracts
SharedCommon provides framework-neutral permission mapping contracts under:
Maatify\SharedCommon\Contracts\Security
These contracts allow independent Maatify modules to expose route-to-permission requirements without depending on AdminKernel or any application-specific security implementation.
This keeps modules reusable and decoupled while allowing the application or kernel layer to aggregate permission maps and convert them into its own authorization model.
Defining Permission Requirements
Use PermissionRequirementDefinition to describe the permission requirement for a route.
use Maatify\SharedCommon\Contracts\Security\PermissionRequirementDefinition; $single = PermissionRequirementDefinition::single('payment_methods.list'); $anyOf = PermissionRequirementDefinition::anyOf([ 'payment_methods.list', 'payment_methods.dropdown', ]); $allOf = PermissionRequirementDefinition::allOf([ 'payment_methods.update', 'payment_methods.translations.upsert', ]); $compound = PermissionRequirementDefinition::compound( anyOf: ['payment_methods.list', 'payment_methods.dropdown'], allOf: ['admin.access'], );
Providing a Permission Map from a Module
A module can expose its route permission map by implementing PermissionMapProviderInterface.
<?php declare(strict_types=1); namespace Maatify\PaymentMethod\Security; use Maatify\SharedCommon\Contracts\Security\PermissionMapProviderInterface; use Maatify\SharedCommon\Contracts\Security\PermissionRequirementDefinition; final class PaymentMethodPermissionMapProvider implements PermissionMapProviderInterface { /** * @return array<string, PermissionRequirementDefinition> */ public function permissionMap(): array { return [ 'payment_methods.list.ui' => PermissionRequirementDefinition::single('payment_methods.list'), 'payment_methods.list.api' => PermissionRequirementDefinition::single('payment_methods.list'), 'payment_methods.dropdown.api' => PermissionRequirementDefinition::anyOf([ 'payment_methods.list', 'payment_methods.dropdown', ]), 'payment_methods.create.api' => PermissionRequirementDefinition::single('payment_methods.create'), 'payment_methods.update.api' => PermissionRequirementDefinition::single('payment_methods.update'), ]; } }
Exposing Permission Map Providers from a Package
A package or module-level entry point may implement ProvidesPermissionMapsInterface to expose one or more permission map providers.
<?php declare(strict_types=1); namespace Maatify\PaymentMethod; use Maatify\PaymentMethod\Security\PaymentMethodPermissionMapProvider; use Maatify\SharedCommon\Contracts\Security\PermissionMapProviderInterface; use Maatify\SharedCommon\Contracts\Security\ProvidesPermissionMapsInterface; final class PaymentMethodPackage implements ProvidesPermissionMapsInterface { /** * @return list<PermissionMapProviderInterface> */ public function permissionMapProviders(): array { return [ new PaymentMethodPermissionMapProvider(), ]; } }
The consuming application or kernel layer is responsible for collecting these providers and converting the neutral definitions into its own authorization objects.
Further Documentation
- How to Use - Practical integration instructions.
- Changelog - History and evolution of the module.
Documentation Book
Comprehensive architecture and integration guides are available in the Book:
| Chapter | Description |
|---|---|
| Table of Contents | Main entry point for the documentation book. |
| 01. Overview | The philosophy and purpose behind SharedCommon. |
| 02. Architecture | Layering and separation of interfaces and infrastructure. |
| 03. Domain Objects | Core contracts representing the system state context. |
| 04. Clock Abstraction | Why the ClockInterface is crucial for testability and consistency. |
| 05. Integration Patterns | Real-world DI container wiring and cross-module usage. |
| 06. Extension Points | How to provide application-specific context implementations. |