用 Docker 搭建 Laravel 开发环境

在这篇文章中我们将通过Docker在个人本地电脑上构建一个快速、轻量级、不依赖本地电脑所安装的任何开发套件的可复制的Laravel和Vue项目的开发环境(开发环境的所有依赖都安装在Docker构建容器里),加入Vue只是因为有的项目里会在Laravel项目中使用Vue做前后端分离开发,开发环境中需要安装前端开发需要的工具集,当然前后端也可以分成两个项目开发,这个话题不在本篇文章的讨论范围内。

所以我们的目标是:

  • 不在本地安装Mamp/Wamp这样的软件
  • 不使用类似Vagrant这样的虚拟机
  • 不在本地电脑全局安装PHP开发所需要的工具集
  • 不在本地电脑全局安装前端开发所需要的工具集
  • 不在本地电脑全局安装Mysql和Nginx

开始前你需要先去安装一个Docker客户端,Docker的官网中有详细的安装方法。

第一步:获取Laravel的源码包

因为我们电脑上不安装Composer,所以就不能使用Composer来创建Laravel项目了, 这里我使用cURL直接从github上下载了最新的Laravel源码包,你也可以使用wget或者git clone 来获取源码包。

curl -L -O  https://github.com/laravel/laravel/archive/v5.5.0.tar.gz /
&& tar -zxvf v5.5.0.tar.gz /
&& rm v5.5.0.tar.gz

上面的命令在curl下载完源码包后会解压源码压缩包,解压完成后在把源码压缩包v5.5.0.tar.gz删掉,执行完后你会看到一个laravel-5.5.0的项目目录。

第二步:添加docker-compose.yml

在项目中创建docker-compose.yml文件。

Compose 项目是 Docker 官方的开源项目,负责实现对 Docker 容器集群的快速编排。我们知道使用一个 Dockerfile 模板文件,可以让用户很方便的定义一个单独的应用容器。在这里我们会用到四个容器分别将PHPMysqlNginx放在四个不同的容器中,通过compose将四个应用容器关联到一起组成项目。

编排文件的开头如下:

version: '2'
services:
   # our services will go here

在编排文件中,把每个容器叫做一个服务,services下定义整个应用中用到的所有服务(即容器)。

App服务

APP服务的容器将执行我们项目中的代码。

app:
  build:
    context: ./
    dockerfile: app.dockerfile
  working_dir: /var/www
  volumes:
    - ./:/var/www
  environment:
    - "DB_PORT=3306"
    - "DB_HOST=database"

Notes:

  • 我们使用app.dockerfile这个镜像文件来构建我们的App容器,在镜像文件中我们会对项目中用到的PHP模块镜像配置,也会额外安装NPM用来构建前端代码。
  • working_dir: /var/www把工作目录设置成了/var/www,在容器中项目代码将会被放在/var/www目录下面,包括使用docker exec app执行的命令也都是以/var/www为当前工作目录的。
  • volumes是容器内数据卷所挂载路径设置,在这里我们只定义一个数据卷,把宿主机项目目录挂到在容器中的/var/www上,这样我们在本地电脑对项目代码进行的更改就会马上同步到容器中去,反过来也是一样,容器中对代码做的更改也会及时反馈到本地电脑的项目中。
  • environment设置环境变量名,这里我们设置了DB_PORTDB_HOST 这样就不用修改项目中的.env文件里关于这两项的值了,当然任何你需要在开发环境单独设置的环境变量都可以写到这里,Laravel读取配置使用的DotEnv会检测是否系统有指定环境变量的设置,有的话就不会在去读取.env文件了。

现在我们需要创建上面build环节中提到的app.dockerfile这个文件了,具体内容如下:

FROM php:7.1.22-fpm

# Update packages
RUN apt-get update

# Install PHP and composer dependencies
RUN apt-get install -qq git curl libmcrypt-dev libjpeg-dev libpng-dev libfreetype6-dev libbz2-dev

# Clear out the local repository of retrieved package files
# RUN apt-get clean

# Install needed extensions
# Here you can install any other extension that you need during the test and deployment process
RUN apt-get clean; docker-php-ext-install pdo pdo_mysql mcrypt zip gd pcntl opcache bcmath


# Installs Composer to easily manage your PHP dependencies.
RUN curl --silent --show-error https://getcomposer.org/installer | php -- --install-dir=/usr/local/bin --filename=composer

# Install Node
RUN apt-get update &&\
    apt-get install -y --no-install-recommends gnupg &&\
    curl -sL https://deb.nodesource.com/setup_10.x | bash - &&\
    apt-get update &&\
    apt-get install -y --no-install-recommends nodejs &&\
    npm config set registry https://registry.npm.taobao.org --global &&\
    npm install --global gulp-cli

CMD php-fpm

Notes:

  • 我在这里先将NPM和Composer装到了app容器中,因为在开发时经常需要执行他们,如果发布到生产环境,一般是使用单独的composer对项目代码进行构建而不是放在运行应用的容器里,容器的核心思想之一就是保持单一,这样才能做到快速增加相同角色的容器。

Web服务

接下来,我们需要配置一个Web服务器用,我们把这个容器在编排文件中命名成web

web:
  build:
    context: ./
    dockerfile: web.dockerfile
  working_dir: /var/www
  volumes_from:
    - app
  ports:
    - 8080:80

Notes:

  • volumes_from用来复用在app服务中定义的数据卷路径

  • 通过ports将本地电脑的8080端口映射到web容器的80端口,这样在开发环境中我们就不用设置hosts文件,直接通过IP加端口就能访问服务了。

    Web服务器选用nginx,所以我们需要用一个nginx镜像文件来构建这个容器,在这之前我们需要在nginx镜像的基础上再设置一下项目中用到的vhost,所以我们需要一个web.dockerfile文件,它的定义如下:

FROM nginx:1.10

ADD vhost.conf /etc/nginx/conf.d/default.conf

根据镜像文件的定义,我们把项目中的vhost.conf复制到了容器的/etc/nginx/conf.d/default.conf中,这样基本的nginx配置就配置好了,vhost.conf中的定义如下:

server {
    listen 80;
    index index.php index.html;
    root /var/www/public;

    location / {
        try_files $uri /index.php?$args;
    }

    location ~ \.php$ {
        fastcgi_split_path_info ^(.+\.php)(/.+)$;
        fastcgi_pass app:9000;
        fastcgi_index index.php;
        include fastcgi_params;
        fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
        fastcgi_param PATH_INFO $fastcgi_path_info;
    }
}

Notes:

  • 因为是开发环境我们就只进行最简单的配置,不做调优考虑了。
  • fastcgi_pass app:9000; nginx将对PHP的请求通过fastcgi传递给了app服务的9000端口,docker-compose会自动把services中定义的容器服务连接起来,各个服务相互之间使用服务名称引用。

Mysql服务

接下来我们将配置Mysql服务,与上面两个服务有点不一样的是,在PHP-FPM和Nginx的容器中,我们配置本地电脑的文件可以同步到容器中供容器访问,这让我们开发时对文件作的更改能够快速的在容器中得到反馈加快我们的开发过程。但是在数据库容器中我们希望容器中创建的文件能够持久化(默认容器销毁时,容器内创建的文件也会被销毁),我们可以通过Docker的数据卷来实现上述功能,只不过这次不用再把本地电脑的文件挂在到数据卷上了,Docker客户端会管理创建的数据卷的在本地电脑上具体存储的位置。

下面是编排文件中对database服务的设置


version: '2'
services:

  database:
    image: mysql:5.7
    volumes:
      - dbdata:/var/lib/mysql
    environment:
      - "MYSQL_DATABASE=homestead"
      - "MYSQL_USER=homestead"
      - "MYSQL_PASSWORD=secret"
      - "MYSQL_ROOT_PASSWORD=secret"
    ports:
        - "33061:3306"

volumes:
  dbdata:

Notes:

  • 在文件的最下面我们通过volumes命令创建了一个名为dbdata的数据卷(dbdata后面的冒号是有意写上去的,这是YML文件的一个语法限制,不用太关心)
  • 定义完数据卷后,在上面我们使用<name>:<dir>的格式,通知Docker,将dbdata数据卷挂在到容器中的/var/lib/mysql目录上
  • environments中设置的是Mysql的docker镜像需要的四个必要参数。
  • ports端口映射中,我们将本地电脑的33061端口映射到容器的3306端口,这样我们就能通过电脑上的数据库工具连接到docker内的Mysql了。

将所有服务编排到一起

下面是完整的docker-compose.yml文件,通过编排文件我们将三个应用容器关联在一起组成了项目的服务端

version: '2'
services:

  # The Application
  app:
    build:
      context: ./
      dockerfile: app.dockerfile
    working_dir: /var/www
    volumes:
      - ./:/var/www
    environment:
      - "DB_PORT=3306"
      - "DB_HOST=database"

  # The Web Server
  web:
    build:
      context: ./
      dockerfile: web.dockerfile
    working_dir: /var/www
    volumes_from:
      - app
    ports:
      - 8080:80

  # The Database
  database:
    image: mysql:5.6
    volumes:
      - dbdata:/var/lib/mysql
    environment:
      - "MYSQL_DATABASE=homestead"
      - "MYSQL_USER=homestead"
      - "MYSQL_PASSWORD=secret"
      - "MYSQL_ROOT_PASSWORD=secret"
    ports:
        - "33061:3306"

volumes:
  dbdata:

启动服务

按照上面的步骤配置好编排文件还有指定的docker镜像文件后,我们就可以通过下面的命令启动服务了,执行完后会启动上面文件里定义的三个服务。

docker-compose up -d 

第一次启动时,由于docker客户端要下载上面提到的三个镜像并且构建服务所以启动速度会慢一些,等到下载完镜像并构建完成后,以后的启动都会非常快。

初始化Laravel项目

启动完服务后我们可以初始化Laravel项目了,步骤跟官方文档里介绍的一样,但是需要在启动的app服务的容器里执行:

docker-compose exec app composer install
docker-compose exec app npm install // 如果包含前端项目的话再执行相关命令
docker-compose exec app cp .env.example .env
docker-compose exec app php artisan key:generate
docker-compose exec app php artisan optimize
docker-compose exec app php artisan migrate --seed
docker-compose exec app php artisan make:controller MyController

Notes:

  • docker-compose exec 将命令发送到指定的容器中去执行
  • app是定义在docker-compose.yml中的一个服务,它是一个运行着php-fpm的容器
  • php artisan migrate 是要在容器里执行的命令

查看nginx日志的方法:

  • docker ps 找到nginx服务的container id
  • docker exec -it < contianer id > /bin/bash 进入nginx容器
  • nginx日志的具体路径请查看项目中的vhost.conf

执行完上面的命令后你就能通过http://127.0.0.1:8080/访问到项目啦。

在我的Github gist有一组参考文件方便同学们参考https://gist.github.com/kevinyan815/fa0760...

gist里的文件稍微旧一些,后来在使用的过程中又加入些新的PHP模块和Node,之前composer也单独放到了一个容器中,不过相信聪明的你看到这里应该已经会根据需求更改这些文件啦。

用 Docker 搭建 Laravel 开发环境

本作品采用《CC 协议》,转载必须注明作者和本文链接
公众号:网管叨bi叨 | Golang、Laravel、Docker、K8s等学习经验分享
本帖由系统于 4年前 自动加精
《L01 基础入门》
我们将带你从零开发一个项目并部署到线上,本课程教授 Web 开发中专业、实用的技能,如 Git 工作流、Laravel Mix 前端工作流等。
《L04 微信小程序从零到发布》
从小程序个人账户申请开始,带你一步步进行开发一个微信小程序,直到提交微信控制台上线发布。
讨论数量: 24

在 「第一步」中 curl 的是v5.5.0.tar.gz 版本, 后面的确是 v5.8.0.tar.gz :joy::joy:

5年前 评论

很详细,学习了 :grimacing:

5年前 评论

生产环境来一发@KevinYan

5年前 评论

@KevinYan 我要是多个站点同时运行的话?运行完第一个项目,第二个项目只建web容器?

5年前 评论

@zhaomengqiao 你是想多个PHP版本吗

5年前 评论

生产环境压根没法用 Docker Compose... 尤其是还要求多个站点,还是 Kubernetes 靠谱。

5年前 评论

这里我使用cURL直接从github上下载了最新的Laravel源码包

最新是 5.8。

5年前 评论
Bin

讲的很详细,希望能把多个项目,跟多个 PHP版本切换,然后关于 docker 在 phpsorm 下 debug调试也加上, 就更完美了 。我是不是要求的有点高?哈哈

4年前 评论

我个人遇到的主要问题是从 homestead 迁移到 docker 环境不习惯的问题.

前期主要的问题是:

  • homestead 挂载共享目录 (工作目录) 很方便, 但 docker 一开始不知道也可以挂载共享目录
  • docker 制作镜像不知从何入手, 而且一开始也没整明白配置文件的挂载, 例如 nginx 的站点目录

参考了 laradock, 以及了解了一下 docker-compose, 知道了这些特性可以很方便解决从 homestead 迁移过去的忧虑:

  • 大部分镜像已经做到开箱即用, 只需要配置好环境变量即可
  • 可以挂载共享目录. 也就是除了工作目录以外, nginx 的配置文件, php 的配置 (php.ini/php-fpm.conf) 放在宿主机, 然后挂载到容器中, 而不是在制作镜像的时候复制到镜像里.
  • 端口映射跟 homestead 一样, 但在 docker-compose 中可以配置一段范围的端口映射, 对我的开发习惯很方便.
  • 使用 docker-compose 时, 容器间互相访问是以 service name 作为 hostname, 非常方便. 在 nginx 的容器里要调用 fpm 只需要在 nginx 里配置 php:9000 即可.
  • docker-compose 支持环境变量, 在 php.ini 等的配置里也支持使用环境变量.
  • docker-compose 还支持多配置文件 override 的功能

那到了这样, 起一个 nginx 作为例子

version: '3'
services:
    nginx:
        image: nginx:alpine
        ports:
            - "80:80"
            # 我的个人习惯是本地开发不改 host, 直接开端口, 所以划分了一个端口范围
            - "9301-9309:9301-9309"    
        volumes:
            - "/path/to/your/workspace:/home/www-data"
            # 把宿主机的 nginx.conf 挂到容器里, 比 homestead 还方便 
            - "/path/to/your/nginx.conf:/etc/nginx/nginx.conf" 
            # 同理, 把 nginx 所有的 server 配置从宿主机挂到容器里
            - "/path/to/your/vhost/config/:/etc/nginx/sites" 

启动一下

docker-composer up -d nginx

然后访问以下 localhost, 随便改一下 nginx 配置然后执行一下

docker exec -i xxxxxxxxx nginx -s reload

那这样基本就理清了思路, 所以抄了 laradock 搞了一个

https://github.com/RunnerLee/runnerdock

使用起来也很方便, 把仓库拉到本地, 然后复制 env-example.env, 并在里面配置一下工作目录, 日志文件的挂载地址, xdebug 的一些配置, 然后就跑起来了

4年前 评论

@Bin 其实我一直不太明白为什么一直有多个 php 版本切换的需求...

我需要同时使用 php5.6 跟 php7.2, 所以我就两个 fpm 都开着, 终端里默认 php 版本是 7.2, 给 5.6 的php 设置了个别名. 两个版本并存.

4年前 评论
loveinalife 3年前

为什么没有一个 download && click and run 的操作,我为什么要知道这么多细节,我只是想开发 laravel 而已 --

4年前 评论

@sniffrose 借用 2gua 大佬今天在微博的一句话, 简单不等于容易. 大概懂得原理跟方法的话, 用 docker 来做开发环境是非常简单便利的. 但刚开始肯定不容易, 容易一般代表了价值不高.

4年前 评论

加入 CI 这样就不用进容器手动敲那几个命令了

4年前 评论

FastCGI sent in stderr: "Primary script unknown" while reading response header from upstream

按文搭建 请求到了nginx后报错 有大佬遇到过吗

4年前 评论

请问如果两个laravel项目,版本一样。是都放在www下面。还是再来这么一套。还有生成环境就是这么部署的吗?

4年前 评论
KevinYan (楼主) 4年前
congcong (作者) 4年前

大佬,用docker-compose exec app php artisan make:controller MyController 创建的控制器是root用户所有,要怎么改成别的用户

4年前 评论
KevinYan (楼主) 4年前
TigerLin

最近在使用的deepin 系统 学习了解docker,有个疑问能否请教下,laravel 环境的部署 是在docker里先pull centos系统,然后在这个centos容器内 在安装mysql、php、nginx来部署laravel ,还是直接单独pull安装php、mysql、laravel呢,作为新手没看能看懂这篇文章,能否指导一下,缺乏一个系统的认知

4年前 评论
KevinYan (楼主) 4年前
TigerLin (作者) 4年前
KevinYan (楼主) 4年前
TigerLin (作者) 4年前

@深入浅出 因为每个 docker 镜像都是在基础镜像的基础上构建出来,基础镜像一般都是最小的 linux 系统,剥离了一切没必要的东西,里面甚至没有 vim. 你最好跟着Docker 官网的入门教程学,感觉有点偏了。镜像的基础原理看 http://dwz.date/4VG

4年前 评论

我有一个问题,实际开发过程中如果需要安装新的扩展怎么办? 改写Dockerfile重新build一下吗?每次build镜像的过程,执行一堆命令,感觉太浪费时间了。

4年前 评论

@xiangxihenli 麻烦只是一个主观感受

4年前 评论

Step 4/8 : RUN apt-get install -qq git curl libmcrypt-dev libjpeg-dev libpng-dev libfreetype6-dev libbz2-dev ---> Running in d453e662bae3 E: Unable to correct problems, you have held broken packages. ERROR: Service 'app' failed to build: The command '/bin/sh -c apt-get install -qq git curl libmcrypt-dev libjpeg-dev libpng-dev libfreetype6-dev libbz2-dev' returned a non-zero code: 100 请教下我这个问题该如何解决呢,尝试卸载docker再重装都无法解决

3年前 评论
KevinYan (楼主) 3年前

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!
未填写
文章
113
粉丝
365
喜欢
484
收藏
314
排名:34
访问:20.4 万
私信
所有博文
社区赞助商