f4php/framework

F4 is a lightweight web development framework

v0.1.10 2025-07-05 18:25 UTC

README

This is F4, a lightweight PHP/PostgreSQL-based web application development framework.

Quick start

To quickly start a new application, you should refer to f4php/f4 package, which works as a self-documented skeletal application.

In order to create your own project using composer, use the following command:

composer create-project f4php/f4

and then follow official documentation.

Developer's guide

Introduction

F4 is a very lightweight framework, it provides a developer with minimal web app infrastructure and tries its best to let you focus on application-specific logic.

Architecture

F4 framework architecture includes several key elements, but most of the developer-facing features are documented in f4php/f4. If you just want to try coding with F4, definitely start there.

The main feature that is not documented elsewhere is the bootstrap process, which is described below.

F4\Loader and F4\Core

F4's core philosophy, similar to other PHP framworks, is based on the idea that it controls (or takes over) the processing of any incoming HTTP request, and allows app developer to focus on a (mostly) declarative description of how the to handle the request.

F4 uses its own bootstrap process to initialize itself and collect any app-specific information (coming in the form of Module's') before passing instances of Request and Response to app-specific Route's. It also provides a flexible PostgreSQL-compatible database query builder as a DB class.

Code below is the only part that is placed in a web server root folder (public/index.php by default).

Each bootstrap phase supports custom handlers that allow low-level intervention. It is highly unlikely that your project would ever require this feature, but it's there if you need it.

<?php

declare(strict_types=1);

use F4\Loader;
use F4\Core;

error_reporting(error_level: E_ALL & ~E_DEPRECATED);

require_once __DIR__.'/../vendor/autoload.php';

Loader::setPath(path: __DIR__ . '/../');
// Loader::setAssetPath(path: __DIR__ . '/assets/');
Loader::loadEnvironmentConfig(environments: [($_SERVER['F4_ENVIRONMENT']??null)?:'local', 'default']);

(new Core(/*
        $alternativeCoreApiProxyClassName,
        $alternativeRouterClassName,
        $alternativeDebuggerClassName,
    */))
    ->setUpRequestResponse(
        /*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ->setUpEnvironment(
        /*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ->setUpLocalizer(
        /*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ->setUpEmitter(
        /*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ->registerModules(/*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ->processRequest(/*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ->emitResponse(/*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ->restoreEnvironment(/*
        function($defaultHandler) {
            // This place is for dirty hacks only, please refer to class structure for better customization options
            // $this refers to Core instance
            $defaultHandler();
        }
    */)
    ;

This bootstrap sequence is mostly self-explanatory, and contains the following steps:

  1. Setting Loader base path
  2. Locating and loading main configuration file based on "environment"
    1. The list of environments is defined in composer.json
    2. Selection of a specific environment for each request is based on F4_ENVIRONMENT process environment variable, which may come a bit confusing at first, but this gives DevOps engineers a way to control how F4 boots in a specific environment
    3. Default environments names of 'local' and 'default' are used, in that particular order of preference, if F4_ENVIRONMENT is not set
    4. Failure to locate and load the main configuration file will prevent F4 from bootstrapping, and will result in a fatal error
  3. Creating a new instance of Core class
  4. Instantiating and initializing internal Request and Response objects based on actual request data
  5. Setting up code runtime environment like output buffer caching, custom exception handlers, internal multibyte encoding, timezone etc.
  6. Setting up localizer
  7. Setting up response emitter relevant to the request
  8. Registering modules. All app-specific code is registered at this stage, but isn't run just yet
  9. Request is processed (we can also say 'routed') to app-specific code, executing all applicable middleware and a matching route handler, if any
  10. Response is emitted to the request initiator (normally, a web browser, but running in CLI is also possible)
  11. Code runtime environment is cleaned up

Custom intervention in the bootstrap process is not recommended. All of the developer-facing features needed to handle a web request, such as routing, parameter validation, header management etc. are provided via F4's Core API and can be used inside Modules.

Such an interventions may be required hoewever if, for example, exotic third-party software needs to be initialized / configured at a certain stage and made available to other parts of the code.

Once again, normal operation does not require any developer intervention in the bootstrap process.

Tests and static analysis

F4 framework package contains unit tests and supports static code analysis with phpstan, using the following commands:

composer run test
composer run phpstan