Composer plugin that ensures scripts are always executed within a Docker Compose service.
composer require --dev empaphy/docker-composer
composer config allow-plugins.empaphy/docker-composer trueComposer 2.2 and newer require plugins to be allowed explicitly. Composer 1 ignores
allow-plugins.
Configure the target service in the root package's composer.json:
{
"extra": {
"docker-composer": {
"service": "php",
"mode": "exec",
"compose-files": ["docker-compose.yaml"],
"project-directory": ".",
"workdir": "/usr/src/app",
"exclude": ["host-only-script"],
"service-mapping": {
"php-test": "test",
"php-tools": ["stan", "cs"]
}
}
}
}Configure service for the default redirection target, and configure
service-mapping for scripts that should run in a different service. If neither
resolves the current script, the plugin warns once per Composer run and lets host
scripts run normally.
Supported keys:
service: Docker Compose service used to run Composer scripts.mode:execorrun; defaults toexec.compose-files: one compose file path or a list of compose file paths.project-directory: optional Docker Compose project directory.workdir: optional working directory inside the container.exclude: exact Composer script/event names that should run on the host.service-mapping: Docker Compose service names mapped to one script or a list of scripts.
Unknown keys warn and are ignored. Invalid known values fail before Docker is run.
When Composer dispatches a top-level script on the host, the plugin runs the same script inside the configured service and prevents the host-side script from continuing.
If service-mapping contains the current script name under a service, that service is used
instead of the default service.
In exec mode, the plugin checks whether the service is already running:
docker compose ps --status running --services <service>If the service is not listed, it starts it:
docker compose up -d <service>
docker compose exec <service> composer run-script <script>In run mode, it runs:
docker compose run --rm <service> composer run-script <script>The plugin passes DOCKER_COMPOSER_INSIDE=1 into the container. Composer scripts
then run normally because the plugin detects that Composer is already inside a
container. It also treats /.dockerenv, /run/.containerenv, and common cgroup
markers as container signals.
Set DOCKER_COMPOSER_DISABLE=1 to bypass Docker redirection temporarily.
This plugin redirects Composer scripts, including lifecycle scripts such as
post-install-cmd and custom scripts run through composer run-script.
It does not transparently replace whole Composer commands such as
composer install with docker compose exec php composer install. Composer's
plugin command events do not provide a clean way to run a child command, skip the
host command, and return the child exit code without relying on fragile internals.