Compatible with popular PHP Frameworks such as Laravel & Symfony and their variants.
Typically deployed on Kubernetes.
Production Grade Image for PHP 8.4+ Applications running Nginx + PHP FPM based on PHP & Nginx Official Images, compatible with popular PHP Frameworks such as Laravel & Symfony and their variants.
- This is a pre-configured template image for your PHP Application, and you shall extend and edit it according to your app requirements. See Requirements first, then follow How to use with my project and How to configure image to run my project for setup instructions.
- The Image utilizes multistage builds to create multiple targets optimized for production OR development use cases. Hyper-optimized for caching and build time. See Image Targets and Build Arguments for details.
- Demo applications are available for both frameworks to give you a quick start:
- use
make demo/symfony/setupfor Symfony Demo (latest Symfony 7.x) - use
make demo/laravel/setupfor Laravel (latest Laravel 11.x).
- use
- Designed to run in orchestrated environments like Kubernetes. See How is it deployed? for architecture details.
- Multi-architecture support - native images for AMD64 and ARM64.
- Uses Alpine based images and multistage builds for minimal images. (~135 MB)
- Multi-Container setup with
Nginx&PHP-FPMcommunicating via TCP. - Productions Image that are immutable and fully contained.
- Runs as non-root in both application containers.
- Configured for graceful shutdowns/restarts, zero downtime deployments, auto-healing, and auto-scaling.
- PHP 8.4 optimizations including JIT compilation and OPcache file caching.
- Easily extend the image with extra configuration, and scripts; such as post-build & pre-run scripts.
- Minimal startup time, container almost start serving requests almost instantly.
- Image tries to fail at build time as much as possible by running all sort of checks.
- Ability to run Commands, Consumers and Crons using same image. (No supervisor or crontab)
- Development Image supports mounting code and hot-reloading and XDebug out of the box.
- Cache-friendly mechanism to update OS packages and auto-patch security vulnerabilities (see cache mechanism).
This is a template, it's expected from you to tailor it to your needs. And then generate a build pipeline to build the image and push it to your registry.
- Copy this repository
Dockerfile,dockerDirectory,Makefile,docker-compose.yml,docker-compose.prod.ymland.dockerignoreto your application root directory and configure it to your needs.
You'll need to iterate over your application's dependency system packages, and required PHP Extensions; and add them to their respective locations in the image.
- Add System Dependencies and PHP Extensions your application depends on to the Image.
- Port in any configuration changes you made for PHP.ini to the image, otherwise use the sane defaults.
make build && make upfor development setup,make deployto run the production variant.
These steps explained in details below.
Your application will be split into two components.
- The Webserver -> Server Static Content and proxy dynamic requests to PHP-FPM over TCP, webserver also applies rate limiting, security headers... and whatever it is configured for.
- The PHP Process -> PHP FPM process that will run you PHP Code.
Other type of deployments such as a cron-job, or a supervised consumer can be achieved by overriding the default image CMD.
- Docker 20.10.0 or higher
- Docker Compose V2 (included with Docker Desktop)
- PHP >= 8.2 Application
- Download This Repository
- Copy
Dockerfile,dockerDirectory,Makefile,docker-compose.yml,docker-compose.prod.ymland.dockerignoreInto your Application Repository.
OR
- Modify
Dockerfileto your app needs, and add your app needed OS Packages, and PHP Extensions.- Dockerfile Header has Build Time Arguments, customize it, most notably the
RUNTIME_DEPSargument. - Below in the
baseimage, add the PHP Extensions your application depends on.
- Dockerfile Header has Build Time Arguments, customize it, most notably the
- Run
make upfor development ormake deployfor production.- For Dev:
make upis just an alias fordocker compose up -d - For Dev: Make sure to delete previous
vendordirectory if you had it before. - Docker-Compose will start App container first, and only start Web server when it's ready, on initial install, it might take some time.
- For Dev:
- Go to http://localhost:8080
Makefile is just a wrapper over docker compose commands.
-
The image comes with a handy Makefile to build the image using Docker-Compose files, it's handy when manually building the image for development or in a not-orchestrated docker host. However, in an environment where CI/CD pipelines will build the image, they will need to supply some build-time arguments for the image. (tho defaults exist.)
ARG Description Default PHP_VERSIONPHP Version used in the Image 8.4PHP_ALPINE_VERSIONAlpine Version for the PHP Image 3.21NGINX_VERSIONNginx Version 1.28COMPOSER_VERSIONComposer Version used in Image 2COMPOSER_AUTHA Json Object with Bitbucket or Github token to clone private Repos with composer.
Reference{}XDEBUG_VERSIONXdebug Version to use in Development Image 3.5.0OS_PACKAGE_UPGRADE_TRIGGERCache buster for OS packages. Changing this value triggers a fresh installation and update of all OS packages. See OS Package Cache Busting for details. 1Target Env Desc Size Based On app Production The PHP Application with immutable code/dependencies. By default starts PHP-FPMprocess listening on9000. Command can be extended to run any PHP Consumer/Job, entrypoint will still start the pre-run setup and then run the supplied command.~135mb PHP Official Image (Alpine) web Production The webserver, an Nginx container that is configured to server static content and forward dynamic requests to the PHP-FPM container running in the appimage variant~21mb Nginx Official Image (Alpine) app-dev Development Development PHP Application variant with dependencies inside. Image expects the code to be mounted on /appto support hot-reloading. You need to mount dummy/app/vendorvolume too to avoid code volume to overwrite dependencies already inside the image.~150mb PHP Official Image (Alpine) web-dev Development Development Webserver with the exact configuration as the production configuration. Expects public directory to be mounted at /app/public~21mb Nginx Official Image (Alpine)
-
The image is to be used as a base for your PHP application image, you should modify its Dockerfile to your needs.
- Install System Packages in the following section in the Dockerfile.
- Add OS Packages needed in
RUNTIME_DEPSin Dockerfile.
- Add OS Packages needed in
- Install PHP Extensions In the following section in the Dockerfile.
# ---------------------------------------- Install / Enable PHP Extensions --------------------------------------------- RUN docker-php-ext-install \ opcache \ intl \ pdo_mysql \ # Pecl Extentions RUN pecl install apcu-5.1.20 && docker-php-ext-enable apcu # EX: RUN pecl install memcached && docker-php-ext-enable memcached
- Install System Packages in the following section in the Dockerfile.
At build time, Image will run
composer check-platform-reqsto check that PHP and extensions versions match the platform requirements of the installed packages.
The OS_PACKAGE_UPGRADE_TRIGGER build argument allows you to force a fresh installation and update of all OS packages by changing its value. This is helpful to force a fresh installation (and patch security vulnerabilities) every time you build the image. A common use case is to set it based on date (e.g week of year, month, year, etc.) that will determine how often you want to update the OS packages.
Note: It's a common recommendation to pin package versions. However in practice this is always a cause of hassle because you'll have to manually commit a change for every little dependency update. You can rely on Alpine Linux to maintain backward compatibility as they patch updates. The template rely on trust in Alpine plus all the pre-flight checks to make sure updates are safe to install.
- PHP
baseConfiguration that are common in all environments indocker/php/base-php.iniπ - PHP
prodOnly Configurationdocker/php/php-prod.iniπ - PHP
devOnly Configurationdocker/php/php-dev.iniπ
- PHP FPM Configuration
docker/fpm/*.confπ
- Nginx Configuration
docker/nginx/*.conf && docker/nginx/conf.d/*π
The development image includes XDebug 3.5.0 pre-configured and ready to use. This setup has been tested with PHPStorm.
XDebug is automatically enabled in the app-dev target with the following settings:
- Port:
9000(DBGp protocol) - IDE Key:
kubephp - Mode:
debug - Client Host:
host.docker.internal(automatically resolves to your Docker host)
The configuration file is located at docker/php/dev-xdebug.ini and can be customized if needed.
These configuration should work for most default IDEs setups. Tested with PHPStorm.
In docker/ directory there is post-build-* and pre-run-* scripts that are used to extend the image and add extra behavior.
-
post-buildcommand runs at the end of Image build.Run as the last step during the image build. Are Often framework specific commands that generate optimized builds, generate assets, etc.
-
pre-runcommand runs in runtime before running the container main commandRuns before the container's CMD, but after the composer's post-install and post-autload-dump. Used for commands that needs to run at runtime before the application is started. Often are scripts that depends on other services or runtime parameters.
-
*-basescripts run on bothproductionanddevelopmentimages.
- Your application should log app logs to stdout.. Read about 12factor/logs
- By default,
php-fpmaccess logs are disabled as they're mirrored onnginx, this is so thatphp-fpmimage will contain only application logs written by PHP. - In production, Image contains source-code, however, you must sync both
php-fpmandnginximages so that they contain the same build.
-
Why two containers instead of one ?
- In containerized environment, you need to only run one process inside the container. This allows us to better instrument our application for many reasons like separation of health status, metrics, logs, etc.
-
Image Build Fails as it try to connect to DB.
- A typical scenario in most frameworks that comes with
DoctrineORM is that if Doctrine not configured with a DB Ver esion, will try to access the DB at php's script initialization (even at the post-install cmd's), and it will fail when it cannot connect to DB. Make sure you configure doctrine to avoid this extra DB Check connection.
- A typical scenario in most frameworks that comes with
This repository includes demo applications for both Symfony and Laravel to show you how an application is expected to be used with this template.
The Symfony Demo application is a full-featured demo that showcases best practices for Symfony development.
Quick Start:
make demo/symfony/setup # Set up the demo
make demo/symfony/up # Start in dev mode
# Visit http://localhost:8080Setup Details
# Download and set up the Symfony demo application
make demo/symfony/setupThis will download the latest Symfony Demo application, install dependencies, and prepare it for use.
Running Options
# Start in development mode (with hot-reloading)
make demo/symfony/up
# Start in production mode (optimized)
make demo/symfony/deployVisit http://localhost:8080 to see the Symfony demo app.
Cleanup
# Remove the Symfony demo app
make demo/symfony/cleanThe Laravel application is the official Laravel framework skeleton with all the features you need to get started.
Quick Start:
make demo/laravel/setup # Set up the demo
make demo/laravel/up # Start in dev mode
# Visit http://localhost:8080Setup Details
# Download and set up the Laravel application
make demo/laravel/setupThis will:
- Download the latest Laravel application
- Install composer dependencies
- Set up the
.envfile - Generate the application encryption key
- Create the SQLite database file
- Run database migrations
Running Options
# Start in development mode (with hot-reloading)
make demo/laravel/up
# Start in production mode (optimized)
make demo/laravel/deployVisit http://localhost:8080 to see the Laravel application.
Cleanup
# Remove the Laravel demo app
make demo/laravel/cleanAdditional Commands
While the demo applications are running, you can use these commands:
# View container logs
make logs
# Execute artisan/console commands
make command COMMAND="php artisan migrate" # Laravel
make command COMMAND="php bin/console cache:clear" # Symfony
# Access container shell
make shell
# Stop the containers
make downImportant Notes
- Both demos use the same
./appdirectory. To switch between frameworks, clean the current demo first. - The Laravel demo uses SQLite by default (database file at
app/database/database.sqlite). - In development mode, code changes are hot-reloaded automatically.
- In production mode, applications are optimized with cached config, routes, views, and events.
MIT License Copyright (c) 2022-2026 Sherif Abdel-Naby
PR(s) are Open and welcomed.

