ntavelis/dockposer

Composer plugin that automatically generates and keeps in sync docker files with php extensions declared in composer.json.

Installs: 159

Dependents: 0

Suggesters: 0

Security: 0

Stars: 12

Watchers: 3

Forks: 3

Open Issues: 1

Type:composer-plugin

v1.1.0 2022-02-21 07:36 UTC

README

CI codecov

Dockposer

Dockposer is a composer plugin that automatically generates and keeps in sync docker files with php extensions declared in composer.json.

The goal of this composer plugin is to keep in sync the php extensions/dependencies that you have declared in the require section of your composer.json file, with your docker files. When you declare a dependency, e.g ext-amqp, in your composer.json this package will automatically generate the required docker instructions to install pecl amqp extension in your docker images. Furthermore this package will sync the required php version with the version of the official docker image that gets pulled.

It is considered a good practice to declare your system's dependencies in the composer.json file, quote from getcomposer.org:

ext- allows you to require PHP extensions (includes core extensions). Versioning can be quite inconsistent here, so it's often a good idea to set the constraint to *. An example of an extension package name is ext-gd

With this composer-plugin by declaring the php extensions in your composer.json, you get the additional benefit of keeping in sync the required extensions with your dockerfile definitions.

Prerequisites

The package will generate docker and docker-compose files. So I assume you have them installed in your system. Composer version 2 and above to be installed to your computer. If you need composer 1 support, please use version 0.3.0 of this package.

Installation

  1. Require the plugin.
    composer require ntavelis/dockposer --dev
  2. Run composer install to trigger the package to run.
  3. Run docker-compose up to initialize and run the docker containers.
  4. Open http://localhost in your browser to see the front page of your application.

Note: The package keeps checking your dependencies, if you notice with your vcs (git) that it updated any docker files, you need to rebuild the docker images by running docker-compose up --build.

How to add php-extensions

To avoid inconsistencies between the installed php extensions, and the ones that are locked in the composer.lock, the following approach is being recommended.

Let's say we want to add these dependencies to our system ext-gmp and ext-bcmath, the recommended procedure is:

  1. Add the extensions to require section of composer.json
    "require": {
            "ext-bcmath": "*",
            "ext-gmp": "*",
        }
  2. Run docker-compose run --rm php-fpm composer install to trigger the package to run, this will generate the required instructions, inside the docker files, to install the php extensions.
  3. Kill any running containers with docker-compose down
  4. Run docker-compose up --build to rebuild the docker images, this will build the php image with the ext-gmp, ext-bcmath included.
  5. Run docker-compose run --rm php-fpm composer update ext-gmp ext-bcmath, to update the composer.lock file. It is worth noticing here that we are executing the composer update from inside the php-fpm container. This is considered good practice, because it avoids the problem of getting a composer error for missing extensions in our local system.

By following the above approach we replace the images with images that include our new required php extensions, in this example ext-gmp and ext-bcmath extensions. Additionally we update the composer.lock file by running the composer update command, from the php-fpm container.

How it works

This package gets triggered every time after the composer tries to resolve your dependencies, e.g composer install or composer update will trigger this package to run.

Dockposer will read the require section of your composer.json file and will generate docker instructions to install the php extensions you have declared in your require section. The php extentions have the ext prefix, e.g. ext-amqp, ext-ldap. It will also read your declared php version in the require section of composer.json and will generate FROM instruction in docker file to pull the official docker image with the same version. If the php version is not declared in the composer.json, it will default to the php version that executes the composer.

It does not only generate the instructions, but it keeps them in sync. Any given time the require section of composer.json will be in sync with the docker instructions, leaving you only to think how you will create your app.

First execution

The first time you will run this package it will generate the below files:

YOU CAN modify any file, dockposer is smart enough to update the contents of the docker files inside the marked regions.

Syncing of dependencies

Every time you run composer install or composer update dockposer will look your require section and if needed it will take action to sync the docker files.

If there was an update in the docker files, do not forget to run docker-compose up --build to rebuild your images, so that they include the dependencies you required.

Marked regions

As mentioned above, dockposer updates the contents of the marked regions. A marked region starts and ends with this type of comment. ###> %s ###.

For example if you have declared as a dependency, ext-ldap inside the composer.json file, dockposer will generate this section in ./docker/php-fpm/Dockerfile

###> ntavelis/dockposer/php-extensions ###
COPY --from=mlocati/php-extension-installer /usr/bin/install-php-extensions /usr/bin/
RUN install-php-extensions \
	ldap
###> ntavelis/dockposer/php-extensions ###

As you keep adding or removing php extensions as dependencies in your composer.json file, you will see that this section gets automatically updated, to be in sync with your dependencies. Of course dockposer doesn't want to get in your way, so you are free to modify the Dockerfile outside the marked regions. If you do not want dockposer to maintain a section, simply delete it, dockposer will figure it out and will not maintain the given region, which means of course that you will lose some functionality.

Configuration

You can configure the names of the files, and the directory names, currently the below configuration is being supported by dockposer.

In order to change the configuration you need to add it in the composer.json under extra key. Example:

...
   "extra": {
        "dockposer": {
            "docker_dir": "docker_rules"
        }
    },
...

Note: When changing a folder, or a file name, all the references to it will use the name you specified.

List of supported extensions

Under the hood in the dockerfile definitions we use the shell scripts from mlocati/docker-php-extension-installer repository.

Please refer here for a list of supported php extensions you can install.

If an extension is pre-installed inside the official docker images, and you declare it in the composer.json, we will not generate any docker instructions for the given library.

Pre-installed php extensions inside official php images:

  • core
  • ctype
  • curl
  • date
  • dom
  • fileinfo
  • filter
  • ftp
  • hash
  • iconv
  • json
  • libxml
  • mbstring
  • mysqlnd
  • openssl
  • pcre
  • pdo
  • pdo_sqlite
  • phar
  • posix
  • readline
  • reflection
  • session
  • simplexml
  • sodium
  • spl
  • sqlite3
  • standard
  • tokenizer
  • xml
  • xmlreader
  • xmlwriter
  • zlib

PHP Frameworks

Symfony

For Symfony the official docker image covers you, you just need to follow the installation instructions.

Laravel

For Laravel, you need the bcmath extension that it is NOT pre-installed inside official docker images. Check Laravel requirements in Laravel docs.

  1. Install the package by following the installation instructions.
  2. Add bcmath extention to the require section of composer.json
    "require": {
        "ext-bcmath": "*",
    }
  3. Run composer install to trigger the plugin to run.
  4. Run docker-compose up to initialize containers.

How to debug package with xdebug - For contributing to this repository

This section describes how to setup your local environment for development on this package. As a user of this package you do not need to do anything described in this section.

Environment setup

  1. Given the following phpstorm config: image

    • Note: Here we name the server configuration as application, this name should match the PHP_IDE_CONFIG="serverName=application" in the command for debugging.

    • Note: It is a good idea to uncheck this configuration from phpstorm, settings -> languages & frameworks -> debug -> uncheck: Force break at first line when no path mapping specified.

  2. Create another project on the same level as this package. So you have the below directory structure:

    /home/antavelis/webprojects/dockposer -> clone of the package inside here

    /home/antavelis/webprojects/demoapp -> a php application with a composer.json file

  3. In the demoapp location require the local version of the dockpose package:

    {
        "type": "project",
        "license": "proprietary",
        "require": {
            "php": "^7.2.5",
            "ext-ctype": "*",
            "ext-iconv": "*",
            "ntavelis/dockposer": "dev-master"
        },
        "repositories": [
            {
                "type": "path",
                "url": "../dockposer"
            }
        ]
    }
    • Note: The url path must point to the location of the package. Composer will symlink the package and, it will recognize it as a plugin.
  4. You will need docker and docker-compose, please install them before proceeding.

Execute the debug command

  1. Navigate to the package's root directory, e.g /home/antavelis/webprojects/dockposer
  2. Navigate to docker folder: cd ./docker
  3. Setup a break point in phpstorm, somewhere in the code, e.g. in the class src/DockposerPlugin.php
  4. Execute:
    docker-compose run --rm -e PHP_IDE_CONFIG="serverName=application" -e COMPOSER_ALLOW_XDEBUG=true phpserver-dockposer bash -c 'cd /srv/app/demoapp; php -d xdebug.remote_host=172.17.0.1 /usr/local/bin/composer install' 
    • Note: This environmental variable COMPOSER_ALLOW_XDEBUG=true instructs composer to NOT disable xdebug, so that we can debug our application.