使用GitHub Actions实现前后端CI/CD到云服务器

一、静态站点部署(前端)

如果你要部署到github pages或者你不用SSR(服务端渲染),那就构建(SSG)静态站点

配置

nextjs配置SSG(静态站点)next.config.mjs,其他前端框架如vue\react按照官网打包配置,在本地打包一下看输出的目录是啥

const nextConfig = {output: "export",
};export default nextConfig;

github设置服务器需要的账号密码等信息

在这里插入图片描述

设置工作流

项目根目录创建.github\workflows\main.yml

# 定义一个名字
name: deploy-website
# 指定在main分支push时执行
on:push:branches: [main]
# 指定任务
jobs:npm-build:name: npm-build工作runs-on: ubuntu-latest # 指定操作系统# 步骤steps:# uses指定用什么包# name指定名称# run指定操作- name: checkout codeuses: actions/checkout@v4.1.7- name: install nodejsuses: actions/setup-node@v4.0.3with:node-version: "20.x"- name: instal depsrun: npm install # 安装依赖- name: build apprun: npm run build # 构建应用- name: copy dist file with sshuses: easingthemes/ssh-deploy@main # 每个插件参数不一样,需要看文档# 这个文件会公开env:REMOTE_HOST: ${{ secrets.REMOTE_HOST }} # 指定 SSH 主机REMOTE_USER: ${{ secrets.REMOTE_USER }} # 指定 SSH 用户名SSH_PRIVATE_KEY: ${{ secrets.SERVER_SSH_KEY}} # 指定 SSH 密码ARGS: "-avzr --delete" # 指定 rsync 参数SOURCE: './out' # 源目录,也就是build构建后的目录,如果直接上传整个项目,那打包就没有意义了TARGET: '/home/dist' # 指定远程服务器上的目标目录,需要提前创建

当你push后会自动打包并将打包后的文件复制到服务器

Docker部署Nginx

  • 提前安装Docker,当然你直接装nginx也是可以的

先创建一个容器生成配置文件

docker run --name nginx -p 9000:80 -d nginx

创建目录

mkdir -p /etc/nginx/conf/

复制配置文件到宿主机

docker cp nginx:/etc/nginx/nginx.conf /etc/nginx/conf/nginx.confdocker cp nginx:/etc/nginx/conf.d /etc/nginx/conf/conf.d

删除nginx

docker stop nginxdocker rm nginx

启动

注意最后一个挂载目录挂载index.html所在的目录

docker run \
-p 80:80 \
--name nginx \
--restart always \
-v /etc/nginx/conf/nginx.conf:/etc/nginx/nginx.conf \
-v /etc/nginx/conf/conf.d:/etc/nginx/conf.d \
-v /var/log:/var/log/nginx \
-v /home/dist/out:/usr/share/nginx/html \
-d nginx:latest

更新后内容没有显示需要重启nginx

docker restart nginx

此时访问你的ip端口就能看到页面了,但它是http的,因为它没有SSL证书

使用openssl签名证书

安装openssl

apt install openssl

生成一个 2048 位的 RSA 私钥文件private.key

openssl genrsa -out private.key 2048

生成一个证书签名请求(CSR)文件 cert.csr

openssl req -new -days 3650 -key private.key -out cert.csr

填写一些必要的信息

Country Name (2 letter code) [AU]:CN
State or Province Name (full name) [Some-State]:cci
Locality Name (eg, city) []:cci
Organization Name (eg, company) [Internet Widgits Pty Ltd]:组织名称
Organizational Unit Name (eg, section) []:组织单位名称
Common Name (e.g. server FQDN or YOUR name) []:普通名字
Email Address []:邮件地址Please enter the following 'extra' attributes
to be sent with your certificate request
A challenge password []:密码
An optional company name []:公司名

从一个cert.csr生成一个自签名的证书cacert.pem

openssl x509 -req -days 3650 -in cert.csr -out cacert.pem -signkey private.key

此时可以看到三个文件

  • cacert.pem:私钥文件,私钥用于对数据进行签名,确保数据的完整性和真实性,并解密由对应的公钥加密的数据
  • cert.csr:证书签名请求文件,包含了你的公钥和关于你身份的信息,如组织名称、通用名、国家等
  • private.key:根证书文件或 CA 证书文件
ls

更改nginx配置

vim /etc/nginx/conf/conf.d/default.conf

修改端口443 ssl,证书文件和私钥文件位置,其他的遵循openssl规范即可,可以使用这里给出的配置

每一个server块都是一个虚拟主机,你可以用多个server块在一台服务器上部署多个网站

server{listen      80;server_name xxx.xyz www.xxxx.xyz xxx.xxx.xyz;# 填你的域名或IPreturn 301 https://$server_name$request_uri;# 将所有80的http请求重定向到443https}
server {listen       443 ssl;server_name  localhost;# 证书文件ssl_certificate     /etc/nginx/cacert.pem;# 私钥文件ssl_certificate_key /etc/nginx/private.key;# ssl验证ssl_session_timeout 5m;# 安全链接的加密协议ssl_protocols TLSv1 TLSv1.1 TLSv1.2 TLSv1.3;# 加密算法ssl_ciphers EECDH+AESGCM:EDH+AESGCM:AES256+EECDH:AES256+EDH;# 使用服务器首选算法ssl_prefer_server_ciphers   on;location / {root   /usr/share/nginx/html;index  index.html index.htm;}error_page   500 502 503 504  /50x.html;location = /50x.html {root   /usr/share/nginx/html;}
}

删除容器后重新配置

docker stop nginxdocker rm nginx

启动

注意添加了端口和挂载的私钥和证书

docker run \
-p 443:443 \
-p 80:80 \
--restart always \
--name nginx \
-v /etc/nginx/conf/nginx.conf:/etc/nginx/nginx.conf \
-v /etc/nginx/conf/conf.d:/etc/nginx/conf.d \
-v /etc/nginx/conf/cacert.pem:/etc/nginx/cacert.pem \
-v /etc/nginx/conf/private.key:/etc/nginx/private.key \
-v /var/log:/var/log/nginx \
-v /home/dist/out:/usr/share/nginx/html \
-d nginx:latest

此时变成了https,由于我们是自己签名的所以仍然会提示不安全,但我们的传输协议已经变成了https,会对数据包进行加密

想不提示就得充值
在这里插入图片描述

可以看到证书信息已经有了
在这里插入图片描述

二、镜像部署(前后通吃)

适合SSR、前端全栈应用,还可以用Docker、k8s部署

这里虽然没有给出后端例子,go、rust、java等等只要按照你在本地部署的流程编写workflows流程即可

  • 1. 编写Dockerfile构建镜像
  • 2. 本地部署
  • 3. 编写workflows:登录Dockerhub、打包镜像、上传镜像、拉取镜像、删除旧容器、部署新容器

环境准备

服务器提前安装Docker

docker --version

创建Docker hub仓库(也可以用阿里云的镜像仓库)

在这里插入图片描述

github actions设置

Dockerhub

在这里插入图片描述

创建token

在这里插入图片描述

github创建对应的密钥信息

在这里插入图片描述

用自己的服务器执行部署流程(github免费每月2000分钟额度)

这里用我们自己的服务器指定github actions后续部署命令,你也可以把打包和执行流程都放在自己服务器上

在这里插入图片描述

如果执行config.sh报错Must not run with sudo

RUNNER_ALLOW_RUNASROOT=true ./config.sh

启动服务

sudo ./svc.sh installsudo ./svc.sh start

此时可以看到服务器已经加入runner了

在这里插入图片描述

在工作流中指定即可用自己的服务器托管

runs-on: self-hosted

配置

项目根目录新建Dockerfile

FROM node:alpine
WORKDIR /app
COPY package*.json ./
RUN npm install --production
COPY . .
RUN npm run build
EXPOSE 3000
CMD ["npm","run", "start"]  
  • FROM 指定基础镜像
  • WORKDIR 设置工作目录
  • RUN 执行shell命令
  • ENV 设置环境变量
  • COPY 复制文件到镜像中
  • USER 指定用户
  • EXPOSE 暴露端口
  • CMD 容器启动时的执行命令,docker run的参数会覆盖CMD指定的参数,多个CMD,只有最后一个CMD才多个
  • ENTRYPOINT 容器启动时的入口点命令,docker run的参数不会覆盖ENTRYPOINT指定的参数

本地部署

先在本地试一下打包正不正常,npm run build,例如把@转换为./,语法降级等操作在这一步先完成,在.next文件夹下就是打包后的文件

docker build -t nextjs-tgm .

创建容器,此时直接访问http://localhost:3000/就可以看到你的项目

docker run -d \
--restart always \
--name nextjs-tgm \
-p 3000:3000 \
nextjs-tgm

停止删除

docker stop nextjs-demodocker rm nextjs-tgm

设置

项目根目录.github\workflows\main.yml

# 定义一个名字
name: deploy-website
# 指定在main分支push时执行
on:push:branches: [main]#     # 你也可以指定打标签的时候触发构建
#     # tags:
#     #   - 'v*' # 当有以 'v' 开头的 tag
# 指定任务
jobs:build:name: build 工作runs-on: ubuntu-latest # 指定操作系统# 步骤steps:# uses指定用什么包# name指定名称# run指定操作- name: checkout codeuses: actions/checkout@v4.1.7- name: Login to Docker Hub # 登录Docker Hubrun: echo "${{ secrets.DOCKERHUB_TOKEN }}" | docker login -u ${{ secrets.DOCKERHUB_USERNAME }} --password-stdin- name: Build and push Docker imagerun: docker build -t vcciccv/nextjs-app . # 指定镜像名字- name: Publish image to docker hubrun: docker push vcciccv/nextjs-app:latest # 推送镜像到Docker Hubdeploy:needs: build # 前一项执行结束再执行这一项runs-on: self-hosted # 指定在自己的服务器上执行steps:- name: Login to Docker Hub # 登录Docker Hubrun: echo "${{ secrets.DOCKERHUB_TOKEN }}" | docker login -u ${{ secrets.DOCKERHUB_USERNAME }} --password-stdin- name: Pull image from docker hubrun: docker pull vcciccv/nextjs-app:latest # 拉取镜像到本地- name: Delete old containerrun: docker rm -f nextjs-app # 删除旧容器- name: Run docker containerrun: docker run -d -p 3000:3000 --name nextjs-app --restart always vcciccv/nextjs-app:latest # 运行镜像

CICD

推送mian分支的代码就可以看到工作流了

在这里插入图片描述

在你的Docker hub也有镜像

在这里插入图片描述

nginx设置

当然你也可以不用nginx,将端口映射到80(http)或443(https),人生苦短何必为难自己,此时通过ip:3000已经可以访问

这时候需要将nginx加入自定义网络,自定义网络可以通过容器名通信

# 列出所有网络
docker network ls
# 创建网络
docker network create pub-network
# 删除网络
docker network rm pub-network
# 将一个容器连接到一个网络
docker network connect pub-network my_container
# 将一个容器从一个网络断开
docker network disconnect pub-network my_container

.github\workflows\main.yml的docker run当中也要加上这段

或者用上面给出的命令直接加入网络,但这里仍然要配置,下次就不用手动加了

--network pub-network

编辑配置文件

请先查看前面的nginx配置

vim /etc/nginx/conf/conf.d/default.conf

设置上游服务器,我们使用的是容器名

upstream nextjs_upstream {server nextjs-app:3000;
}

设置反向代理

    proxy_http_version 1.1;proxy_set_header Upgrade $http_upgrade;proxy_set_header Connection 'upgrade';proxy_set_header Host $host;proxy_cache_bypass $http_upgrade;location / {proxy_pass   http://nextjs_upstream;}

重启

docker restart nginx

现在实现了Nginx反向代理,不用将3000端口暴露给公网了

Nginx优化(前端,可选)

1、 使用Nginx缓存静态资源,这里只给出nextjs的优化

创建缓存目录

mkdir -p /var/cache/nginx

在配置文件最上面添加

proxy_cache_path /var/cache/nginx levels=1:2 keys_zone=STATIC:10m inactive=7d use_temp_path=off;
  • proxy_cache_path:定义缓存的存储路径为 /var/cache/nginx
  • levels=1:2:设置缓存的目录层级结构,这里表示分为两级目录
  • keys_zone=STATIC:10m:创建一个名为 STATIC 的缓存区,大小为 10MB,用于存储缓存的键值对信息
  • inactive=7d:如果缓存项在 7 天内未被访问,则会被清除
  • use_temp_path=off:不使用临时目录来存储缓存数据

设置每个页面的缓存(仅nextjs),npm run build后.next/static是next为每个页面构建的静态资源

    location /_next/static {proxy_cache STATIC;proxy_pass http://nextjs_upstream;}

指定静态资产缓存路径(放在哪里就指定哪里)

    location /public {proxy_cache STATIC;proxy_ignore_headers Cache-Control;proxy_cache_valid 60m;proxy_pass http://nextjs_upstream;}

2、 开启 gzip 压缩
文件在发送给用户之前被压缩

    gzip on;gzip_proxied any;gzip_comp_level 4;gzip_types text/css application/javascript image/svg+xml;
  • 指定压缩的代理条件、压缩级别和要压缩的文件类型

所有配置

server{listen      80;server_name 域名或服务器ip;return 301 https://$server_name$request_uri;}
upstream nextjs_upstream {server nextjs-app:3000;
}
server {listen       443 ssl;server_name  localhost;# 证书文件ssl_certificate     /etc/nginx/cacert.pem;# 私钥文件ssl_certificate_key /etc/nginx/private.key;# ssl验证ssl_session_timeout 5m;# 安全链接的加密协议ssl_protocols TLSv1 TLSv1.1 TLSv1.2 TLSv1.3;# 加密算法ssl_ciphers EECDH+AESGCM:EDH+AESGCM:AES256+EECDH:AES256+EDH;# 使用服务器首选算法ssl_prefer_server_ciphers   on;#charset koi8-r;#access_log  /var/log/nginx/host.access.log  main;gzip on;gzip_proxied any;gzip_comp_level 4;gzip_types text/css application/javascript image/svg+xml;proxy_http_version 1.1;proxy_set_header Upgrade $http_upgrade;proxy_set_header Connection 'upgrade';proxy_set_header Host $host;proxy_cache_bypass $http_upgrade;location /_next/static {proxy_cache STATIC;proxy_pass http://nextjs_upstream;}location /public {proxy_cache STATIC;proxy_ignore_headers Cache-Control;proxy_cache_valid 60m;proxy_pass http://nextjs_upstream;}location / {proxy_pass   http://nextjs_upstream;}error_page   500 502 503 504  /50x.html;location = /50x.html {root   /usr/share/nginx/html;}
}

Nginx还可以配置错误页面,可以用前面给出的SSG的方法生成

其实不想用docker还可以直接用nodejs+nginx,为了节省生命我就不弄了

项目地址:https://github.com/VCCICCV/tgm-front

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/142605.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

跨域训练评估BEVal:自动驾驶 BEV 的跨数据集评估框架

跨域训练评估BEVal:自动驾驶 BEV 的跨数据集评估框架 Abstract 当前在自动驾驶中的鸟瞰图语义分割研究主要集中在使用单个数据集(通常是nuScenes数据集)优化神经网络模型。这种做法导致了高度专业化的模型,可能在面对不同环境或…

孙溟㠭浅析中国碑帖〈曹全碑〉

孙溟㠭浅析中国碑帖《曹全碑》 《曹全碑》 《曹全碑》亦称《郃阳令曹全碑》,东汉时期的碑刻。属于隶书体,东汉中平二年(公元158年)立碑。 《曹全碑》 于明代万历初年在陕西郃阳县莘里村被发现,碑文记载了东汉末年曹全…

2025秋招LLM大模型多模态面试题(七)- 思维链CoT

1.思维链(cot) 论文名称:Chain-of-Thought Prompting Elicits Reasoningin Large Language Models论文连接:Chain-of-Thought Prompting Elicits Reasoningin Large Language Models1.什么是思维链提示? 思维链(CoT)提示过程是一种最近开发的提示方法,它鼓励大语言模型解…

GUI编程14:Icon(图标)、ImageIcon(图像图标)标签

视频链接:16、Icon、ImageIcon标签_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1DJ411B75F?p16&vd_sourceb5775c3a4ea16a5306db9c7c1c1486b5 1.在Label上添加Icon package com.yundait.lesson04;import javax.swing.*; import java.awt.*;public cl…

C++数据结构-树的深度优先搜索及树形模拟法运用(进阶篇)

1. DFS简介 深度优先搜索算法(英语:Depth-First-Search,简称DFS)是一种用于遍历或搜索树或图的算法。 沿着树的深度遍历树的节点,尽可能深的搜索树的分支。当节点v的所在边都己被探寻过或者在搜寻时结点不满足条件&am…

Vue2电商平台项目 (三) Search模块、面包屑(页面自己跳自己)、排序、分页器!

文章目录 一、Search模块1、Search模块的api2、Vuex保存数据3、组件获取vuex数据并渲染(1)、分析请求数据的数据结构(2)、getters简化数据、渲染页面 4、Search模块根据不同的参数获取数据(1)、 派发actions的操作封装为函数(2)、设置带给服务器的参数(3)、Object.assign整理参…

comfyui中报错 Cmd(‘git‘) failed due to: exit code(128) 如何解决

🎈背景 comfyui今天在安装插件的过程中,发现有个插件第一次安装失败后,再次安装就开始报错了,提示: ComfyUI-Inpaint-CropAndStitch install failed: Bad Request 截图如下: 看下后台的报错: …

输入一个整数表示输出函数,也表示组成正方形边的*的数量

//输入一个整数表示输出函数&#xff0c;也表示组成正方形边的*的数量 //空心正方形 #include<stdio.h> int main() {int c 5;int i 0;int a[5][5] { 0 };int j 0;for (i 0; i < c; i){for (j 0; j < c; j){if (i 0)a[i][j] *;else if (j 0)a[i][j] *;el…

python的数据类型详解

python基础 认识python基本类型python的注释风格有三种&#xff08;也可以说是两种&#xff09;python的对齐方式python的多行语句折断字符串类型的“计算”列表的常见用法元组的常见用法集合set的常见用法字典的常见用法bytes类型python的输入输出python中的引用 认识python基…

优化最长上升子序列

前言&#xff1a;平时我们做的题目都是用动态规划做的&#xff0c;但是有没有能够优化一下呢&#xff1f; 有一个结论&#xff0c;长度为 i 的一个序列&#xff0c;最后一个元素一定是构成长度为 i 的序列中最小的 我们可以用二分来优化 题目地址 #include<bits/stdc.h>…

作为HR如何利用好校园招聘的渠道

对于企业来说校招是个非常不错的招聘渠道&#xff0c;虽然应届生没有工作经验&#xff0c;但是有很多具有高潜能的人才就在其中&#xff0c;他们年轻朝气&#xff0c;学习能力强&#xff0c;稍加培养就可以成为得力顺手的能人。 作为HR绝对要善于运营校招的渠道&#xff0c;如…

微服务_1、入门

文章目录 一、 认识微服务二、 微服务演变2.1、 单体架构2.2、 分布式架构2.3、 微服务2.4、 微服务方案对比 三、 注册中心3.1、 Eureka3.2、 Nacos3.2.1、服务分级存储模型3.2.2、权重配置3.2.3、环境隔离 一、 认识微服务 二、 微服务演变 随着互联网行业的发展&#xff0c;…

ICM20948 DMP代码详解(26)

接前一篇文章&#xff1a;ICM20948 DMP代码详解&#xff08;25&#xff09; 上一回解析完了inv_icm20948_load_firmware函数&#xff0c;回到inv_icm20948_initialize_lower_driver函数中&#xff0c;继续往下进行解析。为了便于理解和回顾&#xff0c;再次贴出inv_icm20948_in…

【黑马点评】已解决java.lang.NullPointerException异常

Redis学习Day3——黑马点评项目工程开发-CSDN博客 问题发现及描述 在黑马点评项目中&#xff0c;进行到使用Redis提供的Stream消息队列优化异步秒杀问题时&#xff0c;我在进行jmeter测试时遇到了重大的错误 发现无论怎么测试&#xff0c;一定会进入到catch中&#xff0c;又由…

ST表(算法篇)

算法篇之ST表 引言&#xff1a;ST表实际是一个数据结构&#xff0c;但是它本质是基于dp算法的&#xff0c;而算法题中有时也会用到&#xff0c;这边我就归类于算法篇先把 ST表 概念&#xff1a; ST表适用于解决区间最值的问题(RMQ问题)的数据结构ST表本质是dp算法&#xff…

【工作流集成】springboot+vue工作流审批系统(实际源码)

前言 activiti工作流引擎项目&#xff0c;企业erp、oa、hr、crm等企事业办公系统轻松落地&#xff0c;一套完整并且实际运用在多套项目中的案例&#xff0c;满足日常业务流程审批需求。 一、项目形式 springbootvueactiviti集成了activiti在线编辑器&#xff0c;流行的前后端…

深度揭秘:日志打印的艺术与实战技巧,让你的代码会说话!

&#x1f351;个人主页&#xff1a;Jupiter. &#x1f680; 所属专栏&#xff1a;Linux从入门到进阶 欢迎大家点赞收藏评论&#x1f60a; 目录 &#x1f341;日志&#x1f342;日志分模块实现讲解&#x1f343;日志等级的实现&#x1f965;日志时间*时间的获取* &#x1f308;文…

IntelliJ IDEA 创建 HTML 项目教程

传送门 IntelliJ IDEA 是 JetBrains 提供的一款强大且多功能的集成开发环境&#xff08;IDE&#xff09;&#xff0c;不仅可以用于 Java 开发&#xff0c;还支持多种其他编程语言和技术&#xff0c;包括 HTML、CSS 和 JavaScript 等前端开发工具。本文将带你逐步了解如何使用 …

无人机培训机构必备:驾驶员训练机构合格证技术详解

无人机驾驶员训练机构合格证是针对从事无人机驾驶员培训的机构而设立的资质认证&#xff0c;该证书要求培训机构具备专业的师资力量、完善的教学设施、科学的课程体系以及严格的教学质量监控体系&#xff0c;以确保培训质量和学员安全。以下是对无人机驾驶员训练机构合格证技术…

JavaSE - 面向对象编程02

01 static关键字 01_01 static修饰成员变量 【1】成员变量的分类及特点&#xff1a; ① 类变量&#xff1a;被static修饰&#xff0c;随类一起加载&#xff0c;在计算机中只有一份&#xff0c;被该类的所有对象共享。 ② 实例变量(对象的变量)&#xff1a;不被static修饰&…