本地環境是mac。使用iTerm2+ohMyZsh
服務器環境是阿里云的ecs,綠色純凈centos7一只.
step1. 通過ssh登上服務器。配置免密登錄。
step2. 服務器上安裝docker和php(還有git、composer)。
step3. docker分別配置mysql、php、nginx。目標是能正常訪問phpinfo。
step4. 本地安裝deployer。
step5. git相關,創建一個laravel新項目。
step6. 配置deployer部署項目。目標是能正常訪問laravel歡迎頁以及正常使用laravel orm。
哇哦,看著6步,好像蠻少哦,坑超級多- - 一趟趟坑踩下來,快把自己蠢哭了。
step1. 「ssh登服務器」。
這一步所有操作都在本地,云服務器上不做任何改動。登進去后可輸入exit回車退出,也可ctrl+D退出。
阿里云登上控制臺,云服務器ECS查看示例有個公網ip,就這個公網ip
// 在本地命令行執行
ssh root@{你的云服務器公網 IP}
配置免密登錄,在ECS管理面板中找到密鑰對,點進去,創建密鑰對。
『密鑰對名稱』,我填的是aen233,『創建類型』選擇『自動新建密鑰對』然后點擊『確定』按鈕:頁面會彈出一個下載框下載 aen233.pem 文件,這個文件需要妥善保存,我把它放在~/.ssh下面了,配過git ssh的小伙伴應該都知道這個目錄, 同時 SSH 命令要求密鑰的訪問權限必須是 600。
# 這個我忘記當時走這步沒了,macOS好像木有需要改訪問權限,如果是Linux就需要改權限。
chmod 600 ~/.ssh/aen233.pem
這時候可以免密登錄了,但是要加 -i 參數
$ ssh root@{你的服務器公網 IP} -i ~/.ssh/aen233.pem
其中 -i 參數是告知 SSH 要用后面這個文件作為密鑰登錄。不出意外的話會看到詢問是否信任該服務器,輸入yes,登進。
為了不用每次登錄都得加上 -i 參數, 編輯ssh的配置文件,如果沒有的話,需要新建。
# 本地環境,不是服務器上的
vim ~/.ssh/config
macOS環境我用的atom編輯器。vim或atom都可以啦,總之編輯成下面這樣
Host {你的云服務器公網 IP}
PubkeyAuthentication yes
IdentityFile ~/.ssh/aen233.pem
藍后step1配置完成。
step2. 「服務器上安裝docker和php」。
這一步操作都在云服務器上。
安裝docker
Docker要求64位的系統且內核版本至少為3.10。
執行以下命令。
# 添加yum源。
yum install epel-release –y
yum clean all
yum list
# 安裝并運行Docker。
yum install docker-io –y
systemctl start docker
# 檢查安裝結果。
docker info
安裝php和git、composer
雖然docker也會創建php容器,但是服務器也需要php,而且必須要求php7.1.3+,因為deployer中的映射目錄在服務器,不在docker中,這里就必須要git和composer。
如果php7.0及以下,執行laravel項目的時候,會報laravel require php7.1.3+ 這樣的錯誤。
阿里云的centos7鏡像中,php好像還是5.4還是5.6的版本,要先刪掉,愚蠢如我,先下了7.0的版本,報錯,又刪了7.0的版本,安裝了7.2的版本。
卸載php
yum list installed | grep php
yum remove php*
安裝php7.2
# 首先獲取rpm:
rpm -Uvh https://dl.fedoraproject.org/pub/epel/epel-release-latest-7.noarch.rpm
rpm -Uvh https://mirror.webtatic.com/yum/el7/webtatic-release.rpm
# 然后可以利用 sudo yum list php*查看目前都有php的什么版本了,可以發現從4-7.2的版本都有,7.2版本名為72w,因此安裝該版本即可:
sudo yum -y install php72w
# 但安裝完畢后,輸入php -v發現并沒有該命令,因為php72w只是安裝了php最小的庫,一些應用還未安裝,因此安裝一些拓展包即可:
yum -y install php72w-cli php72w-common php72w-devel php72w-mysql php72w-mbstring
然后是composer
// 下載composer
curl -sS https://getcomposer.org/installer | php
// 將composer.phar文件移動到bin目錄以便全局使用composer命令
mv composer.phar /usr/local/bin/composer
// 切換國內源
composer config -g repo.packagist composer https://packagist.phpcomposer.com
step3. 「docker分別配置mysql、php、nginx」。
雖然有lnmp一鍵包,也有laradock很方便,還是打算分開開,好查錯好維護。
先拉取3個鏡像:這一步不分順序,3個鏡像拉取下來備用
// 獲取mysql鏡像
docker pull mysql:5.7
// 獲取php7.2鏡像
docker pull php:7.2-fpm
// 拉取nginx鏡像
docker pull ngixn:1.12.2
創建3個容器:注意先后順序,先mysql,再php,最后nginx
# 創建mysql容器
docker run -d -p 3306:3306 -v /var/docker/mysql:/var/lib/mysql -e MYSQL_ROOT_PASSWORD=123456 --name aen233_mysql mysql:5.7
/* 參數解釋:
-p:端口映射,映射容器的3306. 例如:3307:3306:將容器的3306端口映射到主機的3307端口
-v:/var/docker/mysql:/var/lib/mysql 將主機目錄/var/docker/mysql掛載到容器的目錄/var/lib/mysql,這個目錄是mysql的數據目錄,如果docker容器刪除,數據還在。
-e:MYSQL_ROOT_PASSWORD=123456 初始化root用戶的密碼
-it:運行交互式的容器,讓docker運行的容器實現"對話"的能力
-d:后臺運行容器,并返回容器ID
--name:命名容器
*/
--------------------------------------------------------
# 創建PHPfpm容器
docker run -d -v /var/www/html:/var/www/html -p 9000:9000 --link aen233_mysql:mysql --name aen233_phpfpm php:7.2-fpm
/* 參數解釋:
-v 前面是主機的目錄:映射容器的目錄,這里需要和nginx容器中一致
--link:掛上msyql。因為php需掛載mysql,所以mysql需要在php之前。
*/
--------------------------------------------------------
# 創建nginx容器
docker run -d -p 80:80 --name aen233_nginx -v /var/www/html:/var/www/html -v /var/docker/nginx/conf.d:/etc/nginx/conf.d --link aen233_phpfpm:phpfpm --name aen233_nginx nginx:1.12.2
/* 參數解釋:
-p:映射80端口
-v:/var/www/html:/var/www/html 這個目錄和php容器中一致。
-v /var/docker/nginx/conf.d:/etc/nginx/conf.d 映射nginx配置目錄,可以不用進docker容器(容器中就不需要安裝vim等),直接在云服務器上修改nginx配置。
--name:容器名
--link:跟PHP關聯,所以nginx容器創建要在php容器創建之后。
*/
接下來是配置docker中的mysql、php、nginx。以及測試目錄映射是否生效。
接下來要進入docker容器了
配置mysql 允許遠程登錄
這樣就可以在本地通過navicat、SequelPro這些數據庫管理工具登進去了
# 進入容器 (可以用name也可以用容器id)
docker exec -it aen233_mysql bash
# 在容器內登陸mysql
root@74a6c7997285:/ mysql -uroot -p
# 為root分配權限,以便可以遠程連接
mysql> grant all PRIVILEGES on *.* to root@'%' WITH GRANT OPTION;
mysql> exit;
配置php的擴展安裝
# 進入到PHP容器 (可以用name也可以用容器id)
docker exec -it aen233_phpfpm /bin/bash
# php安裝pdo_mysql,不走這一步,laravel會報 can not find driver 錯誤
docker-php-ext-install pdo_mysql(curl ...)
# 要安裝php-redis的話,需要另外下載,執行下面這個命令就可以了,有問就no或者空格就好
pecl install redis && docker-php-ext-enable redis
# 安裝后 php-m 發現就都有了哦
劃重點:安裝之后需要docker restart aen233_phpfpm。
劃重點:安裝之后需要docker restart aen233_phpfpm。
劃重點:安裝之后需要docker restart aen233_phpfpm。
配置Nginx容器,讓他支持PHP
先在云服務器修改配置文件default.conf,如果沒有改文件就先創建。
# 這個文件也可以本地寫好,然后通過scp命令傳到云服務器上
vi /var/docker/nginx/conf.d/default.conf
基礎的default.conf如下
server {
listen 80;
server_name localhost;
#charset koi8-r;
access_log /var/log/nginx/host.access.log main;
location / {
root /var/www/html;
index index.php index.html index.htm;
# 如果沒有以下4行,laravel將只能訪問首頁,其他頁面都是404
try_files $uri $uri/ /index.php?$query_string;
if (!-e $request_filename){
rewrite ^/(.*) /index.php last;
}
# 如果沒有以上4行,laravel將只能訪問首頁,其他頁面都是404
}
#error_page 404 /404.html;
# redirect server error pages to the static page /50x.html
#
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root /usr/share/nginx/html;
}
# proxy the PHP scripts to Apache listening on 127.0.0.1:80
#
# location ~ \.php$ {
# proxy_pass http://127.0.0.1;
# }
# pass the PHP scripts to FastCGI server listening on 127.0.0.1:9000
#
location ~ \.php$ {
root /var/www/html;
index index.php index.html;
# 坑在這里,需將原有的127.0.0.1:9000替換成phpfpm:9000
fastcgi_pass phpfpm:9000;
# 坑在這里,需將原有的127.0.0.1:9000替換成phpfpm:9000
fastcgi_index index.php;
# 下面這行也改了 中間的$document_root
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
include fastcgi_params;
}
# deny access to .htaccess files, if Apache's document root
# concurs with nginx's one
#
#location ~ /\.ht {
# deny all;
#}
}
然后進如nginx容器,敲一個命令“nginx -s reload”,退出即可。
# 進入Nginx容器:
docker exec -it aen233_nginx /bin/bash
# 重新加載Nginx
nginx -s reload
測試目錄映射,訪問phpinfo。
先在云服務器的主機 /var/www/html目錄下新建index.php。
<?php
echo phpinfo();
然后瀏覽器輸入你的公網ip,不出意外就是phpinfo頁面啦。
這里要重點去找pdo,查看它的mysql是否是enable狀態。
如果你配php擴展執行了“docker-php-ext-install pdo_mysql”,但是沒有重啟php容器的話,你在容器中php -m,是能查看到pdo_mysql擴展的,但是在phpinfo中是pdo里是沒有mysql的。laravel就會報can not find driver 錯誤。
step4. 「本地安裝deployer」
deployer是安裝在本地的。愚蠢如我,第一遍把它裝在服務器上了
curl -LO https://deployer.org/deployer.phar
mv deployer.phar /usr/local/bin/dep
chmod +x /usr/local/bin/dep
這樣就好了。執行dep 就能看到了
step5.「git相關,創建一個laravel新項目」
首先 要有git賬號、然后要配好ssh key,再然后建立一個倉庫、這個倉庫可以是一個laravel新項目,隨便寫一個路由,控制器的方法中加一行 return User::all();用來測試laravel的orm。
我的項目叫iu。
step6. 「配置deployer部署項目」
創建部署腳本
在本地macOS中創建一個目錄用于放置部署腳本:
我創建的目錄在 ~/Sites/deploy-iu
$ cd ~/Sites/deploy-iu
$ dep init
dep init 命令用來創建一個部署腳本,會詢問我們項目類型,我們是 Laravel 項目所以輸入 1 然后回車;接下來詢問 Repository 也就是我們代碼倉庫的地址,填入自己的 github 倉庫地址即可。
Deployer 在當前目錄下創建了一個名為 deploy.php 的文件。
我最終的deploy.php如下。
<?php
namespace Deployer;
require 'recipe/laravel.php';
set('repository', 'https://github.com/aen233/iu.git');
add('shared_files', []);
add('shared_dirs', []);
add('writable_dirs', []);
// 順便把 composer 的 vendor 目錄也加進來
add('copy_dirs', ['node_modules', 'vendor']);
host('111.22.3.4')
->user('root') // 使用 root 賬號登錄
->identityFile('~/.ssh/aen233.pem') // 指定登錄密鑰文件路徑
->set('deploy_path', '/var/www/html/iu-deployer'); // 指定部署目錄
// 定義一個上傳 .artisan_env 文件的任務
desc('Upload .artisan_env file');
task('artisan_env:upload', function() {
// 將本地的 .env 文件上傳到代碼目錄的 .env
upload('.artisan_env', '{{release_path}}/.env');
});
// 定義一個上傳 .env 文件的任務
desc('Upload .env file');
task('env:upload', function() {
// 將本地的 .env 文件上傳到代碼目錄的 .env
upload('.env', '{{release_path}}/.env');
});
// 在 deploy:vendors 之前調用 deploy:copy_dirs
before('deploy:vendors', 'deploy:copy_dirs');
before('deploy:symlink', 'artisan:migrate');
// 定義一個后置鉤子,在 deploy:shared 之后執行 env:update 任務
after('deploy:shared', 'artisan_env:upload');
after('artisan:migrate', 'env:upload');
after('env:upload', 'artisan:config:cache');
after('artisan:config:cache', 'artisan:route:cache');
after('deploy:failed', 'deploy:unlock');
定義了兩個上傳任務,artisan_env:upload和env:upload,是因為執行migrate時,.env中的DB_HOST=127.0.0.1,但是執行migrate之后,.env的DB_HOST=mysql。
否則laravel會報SQLSTATE[HY000] [2002] Connection refused錯誤。
deploy執行migrate任務時,是在云服務器上操作,但是訪問laravel頁面時,是docker中的nginx和php和mysql交流。每個docker運行的容器都是隔離的,這里的host應該填php容器link的mysql容器。
after('env:upload', 'artisan:config:cache');
在上傳env后,要執行artisan:config:cache。
執行部署命令
藍后 在 ~/Sites/deploy-iu執行:
dep deploy
如果deployer執行composer項目時報錯:
Failed to download laravel/laravel from dist: The zip extension and unzip command are both missing, skipping.
The php.ini used by your command-line PHP is: /etc/php.ini
# 解決辦法: 在服務器上
yum install zip unzip
藍后需要修改下nginx配置
deployer會將current指向當前部署的代碼目錄。
# 將default.conf中的兩處root 改成如下
root /var/www/html/iu-deployer/current/public;
訪問頁面,如果是提示沒有權限。報錯:
The stream or file "/var/www/html/iu/storage/logs/laravel.log" could not be opened: failed to open stream: Permission denied
# 解決辦法:
chmod 777 -R storage/
在iu目錄下,給storage 777權限
還會遇到的報錯,can not find driver,說過了,php容器要有pdo_mysql擴展
SQLSTATE[HY000] [2002] Connection refused,也說過了,.env中的DB_HOST要修改成mysql的docker容器,就是php容器 link的名字。
我自己還遇到一個很想哭的問題,View [welcome] not found. 這個是因為一開始創建容器是,映射的目錄是-v /var/nginx/www/html:/var/www/html,而deployer會把storage共享出來,deployer的配置文件可能就寫死了/var/nginx/www/html,而nginx訪問的目錄是/var/www/html,所以就找不到了。后來我將服務器、php、nginx的映射目錄都統一寫成/var/www/html。就好了。
應該可以咯。
phpstorm code + commit + push。
iterm2 執行命令dep deploy。
自動就發布咯。
參考的教程主要有:
leo 的 電商進階教程 第8章
bestcyt 這篇centos下使用docker搭建lnmp
田勇這篇deployer 實戰經驗分享
謝謝謝謝謝~~~~~