1 云服务器配置
-
购买阿里云的服务器
-
配置ssh密钥对
- 将github的ssh公钥作为密钥对添加至ECS实例中
1
|
$ cat ~/.ssh/id_rsa.pub
|
- 登录至云服务器,第一次使用ssh-copy-id命令访问,输入密码登录,以后就自动使用密钥对登录了
1
2
3
|
$ ssh-copy-id root@47.114.89.76
# 输入密码登录后ctrl d注销登录
$ ssh root@47.114.89.76
|
-
添加安全策略
- 开启测试端口,即供外网访问的端口,如3000、3001、5000、8000、8080,
- 若开启80端口(HTTP默认端口)以及443端口(HTTPS默认端口),则需提前备案
-
云服务器安装docker,复制粘贴下面的命令即可
Install Docker Engine on Ubuntu | Docker Documentation
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
# root用户不需要sudo
$ sudo apt-get update
$ sudo apt-get install \
ca-certificates \
curl \
gnupg \
lsb-release
$ curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo gpg --dearmor -o /usr/share/keyrings/docker-archive-keyring.gpg
$ echo \
"deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/docker-archive-keyring.gpg] https://download.docker.com/linux/ubuntu \
$(lsb_release -cs) stable" | sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
$ sudo apt-get update
$ sudo apt-get install docker-ce docker-ce-cli containerd.io docker-compose-plugin
# 看看是否安装成功
$ docker --version
|
-
推荐只在root用户里安装docker
-
创建新用户,并将其加入docker命令组
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
$ adduser mangosteen
# 输入两次密码,然后回车*n
# 将新用户加入docker命令组
$ usermod -a -G docker mangosteen
# 把root的ssh key及其所有权复制转移到mangosteen用户下
# 创建目录
$ mkdir /home/mangosteen/.ssh
# 复制文件
$ cp ~/.ssh/authorized_keys /home/mangosteen/.ssh
# 转移文件夹权限
$ cd ..
# chown:change own
# -R:递归地
# 用户组:用户
$ chown -R mangosteen:mangosteen .ssh
# 退出,再用新用户登录
$ ssh mangosteen@47.114.89.76
|
2 云服务器部署
2.1 Docker开发环境访问云服务器
- 开发环境生成ssh key
1
|
$ ssh-keygen -t rsa -C "your_email@qq.com"
|
- 开发环境的ssh上传至云服务器的root用户
1
|
$ ssh-copy-id mangosteen@your_ip
|
- 以root用户登录云服务器,将ssh keys赋给mangosteen用户
1
2
3
4
5
6
7
8
9
10
|
$ ssh root@your_ip
# 检查authorized_keys,应该是多了一个
$ cat ~/.ssh/authorized_keys
# 拷贝authorized_keys至mangosteen用户
$ cp ~/.ssh/authorized_keys /home/mangosteen/.ssh
# 权限也移交给mangosteen用户
$ cd /home/mangosteen
$ chown -R mangosteen:mangosteen .ssh
# exit 然后直接使用ssh登录mangosteen用户即可
$ ssh mangosteen@your_ip
|
2.2 打包命令
config/pack_for_remote.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
# 服务器的user和ip
user=mangosteen
ip=47.111.31.50
time=$(date +'%Y%m%d-%H%M%S')
dist=tmp/mangosteen-$time.tar.gz
current_dir=$(dirname $0)
deploy_dir=/home/$user/deploys/$time
gemfile=$current_dir/../Gemfile
gemfile_lock=$current_dir/../Gemfile.lock
function title {
echo
echo "###############################################################################"
echo "## $1"
echo "###############################################################################"
echo
}
yes | rm tmp/mangosteen-*.tar.gz
title '打包源代码为压缩文件'
# 缓存依赖到本地 记得git ignore
bundle cache
tar --exclude="tmp/cache/*" -czv -f $dist *
title '创建远程目录'
ssh $user@$ip "mkdir -p $deploy_dir"
title '上传压缩文件'
# scp = ssh copy
scp $dist $user@$ip:$deploy_dir/
scp $gemfile $user@$ip:$deploy_dir/
scp $gemfile_lock $user@$ip:$deploy_dir/
title '上传 Dockerfile'
scp $current_dir/../config/host.Dockerfile $user@$ip:$deploy_dir/Dockerfile
title '上传 setup 脚本'
scp $current_dir/setup_remote.sh $user@$ip:$deploy_dir/
title '上传版本号'
ssh $user@$ip "echo $time > $deploy_dir/version"
title '执行远程脚本'
ssh $user@$ip "export version=$time; /bin/bash $deploy_dir/setup_remote.sh"
# 要有回车
|
2.3 远程服务器构建脚本
config/setup_remote.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
|
user=mangosteen
root=/home/$user/deploys/$version
container_name=mangosteen-prod-1
db_container_name=db-for-mangosteen
function set_env {
name=$1
while [ -z "${!name}" ]; do
echo "> 请输入 $name:"
read $name
sed -i "1s/^/export $name=${!name}\n/" ~/.bashrc
echo "${name} 已保存至 ~/.bashrc"
done
}
function title {
echo
echo "###############################################################################"
echo "## $1"
echo "###############################################################################"
echo
}
title '设置远程机器的环境变量'
set_env DB_HOST
set_env DB_PASSWORD
set_env RAILS_MASTER_KEY
title '创建数据库'
if [ "$(docker ps -aq -f name=^${DB_HOST}$)" ]; then
echo '已存在数据库'
else
docker run -d --name $DB_HOST \
--network=network1 \
-e POSTGRES_USER=mangosteen \
-e POSTGRES_DB=mangosteen_production \
-e POSTGRES_PASSWORD=$DB_PASSWORD \
-e PGDATA=/var/lib/postgresql/data/pgdata \
-v mangosteen-data:/var/lib/postgresql/data \
postgres:14
echo '创建成功'
fi
title 'docker build'
docker build $root -t mangosteen:$version
if [ "$(docker ps -aq -f name=^mangosteen-prod-1$)" ]; then
title 'docker rm'
docker rm -f $container_name
fi
title 'docker run'
docker run -d -p 3000:3000 \
--network=network1 \
--name=$container_name \
-e DB_HOST=$DB_HOST \
-e DB_PASSWORD=$DB_PASSWORD \
-e RAILS_MASTER_KEY=$RAILS_MASTER_KEY \
mangosteen:$version
echo
echo "是否要更新数据库?[y/N]"
read ans
case $ans in
y|Y|1 ) echo "yes"; title '更新数据库'; docker exec $container_name bin/rails db:create db:migrate ;;
n|N|2 ) echo "no" ;;
"" ) echo "no" ;;
esac
title '全部执行完毕'
|
chmod +x bin/pack_for_remote bin/setup_remote
2.4 host.Dockerfile
config/host.Dockerfile
1
2
3
4
|
...
# 使用bundle cache的包直接安装,无下载过程
RUN bundle install --local
...
|
3 优化bundle install过程
config/host.Dockerfile
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
FROM ruby:3.0.0
ENV RAILS_ENV production
RUN mkdir /mangosteen
RUN bundle config mirror.https://rubygems.org https://gems.ruby-china.com
WORKDIR /mangosteen
# 把Gemfile先拷贝过去,优先进行本地依赖安装
ADD Gemfile /mangosteen
ADD Gemfile.lock /mangosteen
ADD vendor/cache /mangosteen/vendor/cache
RUN bundle config set --local without 'development test'
RUN bundle install --local
# 安装好依赖再把源码拷贝过来,会覆盖掉之前的Gemfile,不过不执行bundle install,所以也无所谓的
ADD mangosteen-*.tar.gz ./
ENTRYPOINT bundle exec puma
|
config/pack_for_remote.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
...
vendor_cache_dir=$current_dir/../vendor/cache
...
title '打包源代码为压缩文件'
# 缓存依赖到本地 记得git ignore
bundle cache
tar --exclude="tmp/cache/*" -czv -f $dist *
...
# 创建deploy_dir以及vendor目录
ssh $user@$ip "mkdir -p $deploy_dir/vendor"
title '上传压缩文件'
...
scp $gemfile_lock $user@$ip:$deploy_dir/
# 将cache也上传到部署目录下 -r表示整个路径下的内容
scp -r $vendor_cache_dir $user@$ip:$deploy_dir/vendor/
title '上传 Dockerfile'
...
|