如何在容器中安装作曲家依赖项?

How to install composer dependencies in a container?

我正在尝试做什么

我正在尝试对基于 CodeIgniter 4 构建的 this php 应用程序进行 docker 化。此应用程序具有以下依赖项,这些依赖项通常通过 composer 安装:

"require": {
    "php": "^7.3 || ^8.0",
    "myth/auth": "dev-develop",
    "codeigniter4/framework": "^4.1",
    "codeigniter4/translations" : "^4.0"
},

项目结构组织如下:

docker
    database
    nginx
    php-fpm
    docker-compose.yml
src
    ... app files ...
tests
composer.json

docker-compose.yml 中,我有一个名为 php-fpm 的服务,它将所有应用程序文件复制到 /var/www/html,即 nginx public 目录:

php-fpm:
  container_name: boilerplate_app
  build:
    context: ./php-fpm
  volumes:
    - ../src:/var/www/html

我需要做的是使用 composer 安装应用程序依赖项。

我试过的

最初,我想在 php-fpm 文件夹中的 Dockerfile 中执行此操作:

FROM php:8.0.2-fpm-alpine

RUN docker-php-ext-install pdo_mysql
RUN docker-php-ext-install mysqli    

# Install Composer
COPY --from=composer /usr/bin/composer /usr/bin/composer

RUN composer require --working-dir=../../ app/boilerplate

COPY ./config/php.ini /usr/local/etc/php/

CMD ["php-fpm"]

EXPOSE 9000

基本上,我已经在容器中安装了 composer,然后我尝试从项目根目录中可用的 composer.json 文件安装所有依赖项。

问题

不幸的是,当我 运行 命令 docker-compose up --build 出现此错误时:

No composer.json present in the current directory (./composer.json), this may be the cause of the following exception.

似乎容器无法访问 composer.json 文件,或者我对路径参数 working-dir.

做了一些不好的事情

因此,我采用了另一种解决方案,在 docker-compose.yml 中定义了另一个容器,其中分隔了 composer 实例:

composer:
  image: composer:2.0.9
  command: ["composer", "require", "app/boilerplate"]
  volumes:
    - /var/www/html

这应该从根目录读取 composer.json 文件,然后应该通过命令 composer require app/boilerplate 安装所有依赖项,但同样会引发相同的错误:

Composer could not find a composer.json file in /app.

这是完整的 docker-compose.yml:

version: '3'

services:
  php-fpm:
    container_name: boilerplate_app
    build:
      context: ./php-fpm
    volumes:
      - ../src:/var/www/html

  nginx:
    container_name: boilerplate_nginx
    build:
      context: ./nginx
    volumes:
      - ../src:/var/www/html
      - ./nginx/nginx.conf:/etc/nginx/nginx.conf
      - ./nginx/sites/:/etc/nginx/sites-available
      - ./nginx/conf.d/:/etc/nginx/conf.d
    ports:
      - "8881:80"
      - "8882:443"
    depends_on:
      - php-fpm

  database:
    container_name: boilerplate_db
    build:
      context: ./database
    environment:
      - MYSQL_DATABASE=boilerplate_db
      - MYSQL_ROOT_PASSWORD=test
      - MYSQL_USER=foo
      - MYSQL_PASSWORD=test
    volumes:
      - ./database/data.sql:/docker-entrypoint-initdb.d/data.sql

  composer:
    image: composer:2.0.9
    command: ["composer", "require", "app/boilerplate"]
    volumes:
      - /var/www/html

这是 composer.json 文件:

{
    "name": "app/boilerplate",
    "description": "CodeIgniter4 Boilerplate based on AdminLTE 3 with user management, roles, permissions, ...",
    "require": {
        "php": "^7.3 || ^8.0",
        "myth/auth": "dev-develop",
        "codeigniter4/framework": "^4.1",
        "codeigniter4/translations" : "^4.0"
    },
    "require-dev": {
        "phpunit/phpunit": "^9.1",
        "fakerphp/faker": "^1.13"
    },
    "minimum-stability": "dev",
    "prefer-stable": true,
    "autoload": {
        "psr-4": {
            "app\boilerplate\": "src"
        }
    },
    "autoload-dev": {
        "psr-4": {
            "Tests\": "tests/"
        }
    },
    "scripts": {
        "post-update-cmd": [
          "@composer dump-autoload"
        ],
        "test": "phpunit --colors=always -vvv"
    }
}

我能做些什么来解决这个问题?

文件夹结构图片预览:

看来你有几个问题。

首先,您依赖于绑定安装。但是这些在 运行-time 可用,但在 build-time.

不可用

因此您需要先将 composer.* 文件复制到 docker 图像,然后再 运行 composer install.

然后,您这样做会遇到麻烦,因为您将构建上下文移到了项目的子目录中,这会使从父目录复制文件时出现问题。

首先,更改构建上下文,然后简单地引用构建项目所需的 Dockerfile

services:
  php-fpm:
    container_name: boilerplate_app
    build:
      dockerfile: ./php-fpm/Dockerfile
      context: .
    volumes:
      - ./src:/var/www/html
# the rest of your docker-compose.yml

执行此操作后,您需要调整 Dockerfile 以说明现在它的构建上下文是一个目录,并添加所需的 COPY 语句。

我还更新了 docker 文件以利用 Docker 多阶段过程

FROM composer:2 as composer_stage

RUN rm -rf /var/www && mkdir -p /var/www/html
WORKDIR /var/www/html

COPY composer.json composer.lock ./

# This are production settings, I'm running with 'no-dev', adjust accordingly 
# if you need it
RUN composer install --ignore-platform-reqs --prefer-dist --no-scripts --no-progress --no-interaction --no-dev --no-autoloader

RUN composer dump-autoload --optimize --apcu --no-dev

FROM php:8.0.2-fpm-alpine

WORKDIR /var/www/html

RUN docker-php-ext-install pdo_mysql
RUN docker-php-ext-install mysqli

COPY php-fpm/config/php.ini /usr/local/etc/php/

COPY --from=composer_stage /var/www/html /var/www/html

COPY src src/

CMD ["php-fpm"]

EXPOSE 9000

我还添加了一个 COPY 语句来复制你的项目 src 目录,否则你正在构建的图像只有你的作曲家依赖关系,没有别的。这样您的应用程序就是一个完全独立的容器(没有双关语意)。

您还应该确保 src 安装点。由于这些并不真正属于图像 building,而是仅在您实际 运行ning 容器时使用,所以我没有不要非常仔细地审查它们。

Dockerize/containerize 应用程序首先是构建正确的图像,然后您需要创建适当的使用文档