ody / framework
ODY framework
Requires
- php: >=8.3
- ody/foundation: ^0.1.0
- ody/server: ^0.1.0
Requires (Dev)
- phpunit/phpunit: ^9.6
- symfony/var-dumper: ^7.2
This package is auto-updated.
Last update: 2025-03-27 22:09:46 UTC
README
Introduction
ODY is a modern PHP API framework built with a focus on high performance and modern architecture. It leverages Swoole's coroutines for asynchronous processing, follows PSR standards for interoperability, and provides a clean architecture for building robust APIs.
Installation
Requirements
- PHP 8.3 or higher
- Swoole PHP extension (≥ 6.0.0)
- Composer
Basic Installation
composer create-project ody/framework your-project-name
cd your-project-name
php ody publish
php ody server:start
Configuration
Configuration files are stored in the config
directory. The primary configuration files include:
app.php
: Application settings, service providers, and middlewaredatabase.php
: Database connections configurationlogging.php
: Logging configuration and channelscache.php
: Cache configuration
Environment-specific configurations can be set in .env
files. A sample .env.example
file is provided that you can copy to .env
and customize:
cp .env.example .env
Project Structure
your-project/
├── app/ # Application code
│ ├── Controllers/ # Controller classes
│ └── ...
├── config/ # Configuration files
├── public/ # Public directory (web server root)
│ └── index.php # Application entry point
├── routes/ # Route definitions
│ └── api.php # API routes
├── src/ # Framework core components
├── storage/ # Storage directory for logs, cache, etc.
├── tests/ # Test files
├── vendor/ # Composer dependencies
├── .env # Environment variables
├── .env.example # Environment variables example
├── composer.json # Composer package file
├── ody # CLI entry point
└── README.md # Project documentation
Routing
Routes are defined in the routes
directory. The framework supports various HTTP methods and route patterns:
// Basic route definition Route::get('/hello', function (ServerRequestInterface $request, ResponseInterface $response) { return $response->json([ 'message' => 'Hello World' ]); }); // Route with named controller Route::post('/users', 'App\Controllers\UserController@store'); // Route with middleware Route::get('/users/{id}', 'App\Controllers\UserController@show') ->middleware('auth'); // Route groups Route::group(['prefix' => '/api/v1', 'middleware' => ['throttle:60,1']], function ($router) { $router->get('/status', function ($request, $response) { return $response->json([ 'status' => 'operational' ]); }); });
Controllers
Controllers handle the application logic and are typically stored in the app/Controllers
directory:
<?php namespace App\Controllers; use Psr\Http\Message\ServerRequestInterface; use Psr\Http\Message\ResponseInterface; use Psr\Log\LoggerInterface; class UserController { private $logger; public function __construct(LoggerInterface $logger) { $this->logger = $logger; } public function index(ServerRequestInterface $request, ResponseInterface $response): ResponseInterface { // Get all users $users = [ ['id' => 1, 'name' => 'John Doe'], ['id' => 2, 'name' => 'Jane Smith'] ]; return $response->withHeader('Content-Type', 'application/json') ->withBody(json_encode($users)); } public function show(ServerRequestInterface $request, ResponseInterface $response, array $params): ResponseInterface { $id = $params['id']; // Get user by ID $user = ['id' => $id, 'name' => 'John Doe']; return $response->withHeader('Content-Type', 'application/json') ->withBody(json_encode($user)); } }
Middleware
Middleware provides a mechanism for filtering and modifying HTTP requests and responses. The ODY Framework implements the PSR-15 middleware standard, allowing for a consistent approach to handling request processing.
Key features:
- PSR-15 compliant implementation
- Support for named middleware
- Middleware grouping
Using Built-in Middleware
Registering Middleware in Routes
You can apply middleware to routes using the middleware()
method:
// Apply a single middleware $router->get('/profile', 'UserController@profile') ->middleware('auth'); // Apply multiple middleware $router->get('/admin/dashboard', 'AdminController@dashboard') ->middleware('auth', 'role:admin');
Global Middleware
Global middleware runs on every request. Configure it in your app.php
configuration file:
'middleware' => [ // Global middleware applied to all routes 'global' => [ Ody\Foundation\Middleware\ErrorHandlerMiddleware::class, Ody\Foundation\Middleware\CorsMiddleware::class, Ody\Foundation\Middleware\JsonBodyParserMiddleware::class, ], ]
Named Middleware
Named middleware allows you to reference middleware by a short name. Define named middleware in your app.php
configuration:
'middleware' => [ // Named middleware that can be referenced in routes 'named' => [ 'auth' => Ody\Foundation\Middleware\AuthMiddleware::class, 'role' => Ody\Foundation\Middleware\RoleMiddleware::class, 'throttle' => Ody\Foundation\Middleware\ThrottleMiddleware::class, 'cors' => Ody\Foundation\Middleware\CorsMiddleware::class, 'json' => Ody\Foundation\Middleware\JsonBodyParserMiddleware::class, ], ]
Middleware Groups
Middleware groups allow you to apply multiple middleware with a single reference. Define groups in your app.php
configuration:
'middleware' => [ // Middleware groups for route groups 'groups' => [ 'web' => [ 'auth', 'json', ], 'api' => [ 'throttle', 'auth', 'json', ], ], ]
Apply a middleware group to a route:
$router->group(['middleware' => 'api'], function ($router) { $router->get('/users', 'UserController@index'); $router->post('/users', 'UserController@store'); });
Defining Middleware with Attributes
You can apply middleware at two levels:
- Controller Level: Middleware applied to all methods in the controller
- Method Level: Middleware applied to specific methods only
Example Controller
<?php namespace App\Controllers; use App\Middleware\RequestLoggerMiddleware; use App\Middleware\AuthMiddleware; use Ody\Foundation\Attributes\Middleware; use Ody\Foundation\Attributes\MiddlewareGroup; use Psr\Http\Message\ServerRequestInterface; use Psr\Http\Message\ResponseInterface; // Apply logging middleware to all methods in this controller #[Middleware(RequestLoggerMiddleware::class)] // Apply a predefined middleware group to all methods #[MiddlewareGroup('api')] class UserController { // This method inherits middleware from the controller class public function index(ServerRequestInterface $request, ResponseInterface $response, array $params): ResponseInterface { // List users... return $response->withJson(['users' => $users]); } // This method has authentication middleware in addition to inherited middleware #[Middleware(AuthMiddleware::class)] public function store(ServerRequestInterface $request, ResponseInterface $response, array $params): ResponseInterface { // Create a user... return $response->withJson(['user' => $newUser]); } // Use parameters with middleware #[Middleware(AuthMiddleware::class, ['guard' => 'admin'])] public function destroy(ServerRequestInterface $request, ResponseInterface $response, array $params): ResponseInterface { // Delete a user... return $response->withJson(['success' => true]); } }
Middleware Attribute Features
Apply a Single Middleware
#[Middleware(MyMiddleware::class)]
Apply Multiple Middleware
Using multiple attributes:
#[Middleware(AuthMiddleware::class)] #[Middleware(ThrottleMiddleware::class)]
Or using an array of middleware:
#[Middleware([LoggingMiddleware::class, CacheMiddleware::class])]
Middleware with Parameters
Pass parameters to middleware:
#[Middleware(ThrottleMiddleware::class, ['maxRequests' => 60, 'minutes' => 1])]
The parameters will be available in the middleware via request attributes:
$maxRequests = $request->getAttribute('middleware_maxRequests');
Middleware Groups
Use predefined middleware groups:
#[MiddlewareGroup('api')]
Creating Custom Middleware
Basic Middleware
Creating a custom middleware requires implementing the PSR-15 MiddlewareInterface
:
<?php namespace App\Http\Middleware; use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; use Psr\Http\Server\MiddlewareInterface; use Psr\Http\Server\RequestHandlerInterface; class CustomMiddleware implements MiddlewareInterface { /** * Process an incoming server request * * @param ServerRequestInterface $request * @param RequestHandlerInterface $handler * @return ResponseInterface */ public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface { // Your logic before passing the request to the next middleware // Process the request with the next middleware or route handler $response = $handler->handle($request); // Your logic after receiving the response from the next middleware return $response; } }
Registering Custom Middleware
Register your custom middleware in the app.php
configuration:
'middleware' => [ 'named' => [ 'custom' => App\Http\Middleware\CustomMiddleware::class, 'custom-param' => App\Http\Middleware\CustomParameterizedMiddleware::class, ], ]
Now you can use your custom middleware in routes:
$router->get('/custom-route', 'Controller@method') ->middleware('custom', 'custom-param:special');
Advanced Usage
Middleware Priority
Middleware executes in the order they are registered. Global middleware runs first, followed by group middleware, and finally route-specific middleware.
Stopping Middleware Execution
To stop the middleware chain and return a response early:
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface { if ($someCondition) { // Return response without calling $handler->handle($request) return new Response() ->withStatus(403) ->json() ->withJson(['error' => 'Access denied']); } return $handler->handle($request); }
Modifying the Request
You can modify the request before passing it to the next middleware:
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface { // Add data to the request $request = $request->withAttribute('custom_data', 'value'); return $handler->handle($request); }
Modifying the Response
You can also modify the response after receiving it from the next middleware:
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface { // Process the request $response = $handler->handle($request); // Modify the response return $response->withHeader('X-Custom-Header', 'value'); }
Dependency Injection
<?php // Binding an interface to a concrete implementation $container->bind(UserRepositoryInterface::class, UserRepository::class); // Registering a singleton $container->singleton(UserService::class, function($container) { return new UserService( $container->make(UserRepositoryInterface::class) ); }); // Resolving dependencies $userService = $container->make(UserService::class);
Logging
Logging is configured in config/logging.php
and provides various channels for logging:
// Using the logger $logger->info('User logged in', ['id' => $userId]); $logger->error('Failed to process payment', ['order_id' => $orderId]); // Using the logger helper function logger()->info('Processing request');
Creating Custom Loggers
Basic Requirements
All custom loggers must:
- Extend
Ody\Logger\AbstractLogger
- Implement a static
create(array $config)
method - Override the
write(string $level, string $message, array $context = [])
method
Example: Creating a Custom Logger
Here's a simple example of a custom logger that logs to Redis:
<?php namespace App\Logging; use Ody\Logger\AbstractLogger; use Ody\Logger\FormatterInterface; use Ody\Logger\JsonFormatter; use Ody\Logger\LineFormatter; use Psr\Log\LoggerInterface; use Psr\Log\LogLevel; use Redis; class RedisLogger extends AbstractLogger { /** * @var Redis */ protected Redis $redis; /** * @var string */ protected string $channel; /** * Constructor */ public function __construct( Redis $redis, string $channel = 'logs', string $level = LogLevel::DEBUG, ?FormatterInterface $formatter = null ) { parent::__construct($level, $formatter); $this->redis = $redis; $this->channel = $channel; } /** * Create a Redis logger from configuration */ public static function create(array $config): LoggerInterface { // Create Redis connection $redis = new Redis(); $redis->connect( $config['host'] ?? '127.0.0.1', $config['port'] ?? 6379 ); if (isset($config['password'])) { $redis->auth($config['password']); } // Create formatter $formatter = null; if (isset($config['formatter'])) { $formatter = self::createFormatter($config); } // Return new logger instance return new self( $redis, $config['channel'] ?? 'logs', $config['level'] ?? LogLevel::DEBUG, $formatter ); } /** * Create a formatter based on configuration */ protected static function createFormatter(array $config): FormatterInterface { $formatterType = $config['formatter'] ?? 'json'; if ($formatterType === 'line') { return new LineFormatter( $config['format'] ?? null, $config['date_format'] ?? null ); } return new JsonFormatter(); } /** * {@inheritdoc} */ protected function write(string $level, string $message, array $context = []): void { // Format log data $logData = [ 'timestamp' => time(), 'level' => $level, 'message' => $message, 'context' => $context ]; // Publish to Redis channel $this->redis->publish( $this->channel, json_encode($logData) ); } }
The create()
Method
The static create()
method is responsible for instantiating your logger based on configuration:
public static function create(array $config): LoggerInterface { // Create dependencies based on configuration // ... // Return new logger instance return new self(...); }
This method receives the channel configuration from the logging.php
config file and should:
- Create any dependencies the logger needs
- Configure those dependencies based on the config array
- Return a new instance of the logger
The write()
Method
The write()
method is where the actual logging happens:
protected function write(string $level, string $message, array $context = []): void { // Implement logging logic here }
This method is called by the parent AbstractLogger
class when a log message needs to be written. It receives:
$level
: The log level (debug, info, warning, etc.)$message
: The formatted log message$context
: Additional context data
Using Custom Loggers
Method 1: Configuration-Based Discovery
The simplest way to use a custom logger is to specify the fully-qualified class name in your logging configuration:
// In config/logging.php 'channels' => [ 'redis' => [ 'driver' => 'redis', 'class' => \App\Logging\RedisLogger::class, 'host' => env('REDIS_HOST', '127.0.0.1'), 'port' => env('REDIS_PORT', 6379), 'channel' => 'application_logs', 'level' => 'debug', ], ]
When you specify a class
parameter, that class will be used regardless of the driver name.
Method 2: Driver Name Registration
You can register your logger with a driver name, which allows you to reference it using just the driver name:
// In a service provider's register method $this->app->make(\Ody\Logger\LogManager::class) ->registerDriver('redis', \App\Logging\RedisLogger::class);
Then in your configuration:
// In config/logging.php 'channels' => [ 'redis' => [ 'driver' => 'redis', // This will use the registered RedisLogger 'host' => env('REDIS_HOST', '127.0.0.1'), 'port' => env('REDIS_PORT', 6379), 'channel' => 'application_logs', 'level' => 'debug', ], ]
Method 3: Automatic Discovery
If your logger follows the naming convention {Driver}Logger
and is in one of the registered namespaces, it will be discovered automatically:
// In config/logging.php 'channels' => [ 'redis' => [ 'driver' => 'redis', // Will look for RedisLogger // Configuration... ], ]
The framework will search for RedisLogger
in the registered namespaces (\Ody\Logger\
and \App\Logging\
by default).
Creating Custom Formatters
If the standard formatters don't meet your needs, you can create your own by implementing the FormatterInterface
:
namespace App\Logging; use Ody\Logger\FormatterInterface; class CustomFormatter implements FormatterInterface { public function format(string $level, string $message, array $context = []): string { // Custom formatting logic return "[$level] $message " . json_encode($context); } }
Service Providers
Service providers are used to register services with the application. Custom service providers can be created in the
app/Providers
directory:
<?php namespace App\Providers; use Ody\Foundation\Providers\ServiceProvider; class CustomServiceProvider extends ServiceProvider { public function register(): void { // Register bindings $this->singleton('custom.service', function() { return new CustomService(); }); } public function boot(): void { // Bootstrap services } }
Register your service provider in config/app.php
:
'providers' => [ // Framework providers Ody\Foundation\Providers\DatabaseServiceProvider::class, // Application providers App\Providers\CustomServiceProvider::class, ],
Running the Application
php ody server:start
Resources
License
ODY Framework is open-source software licensed under the MIT license.