· code · 前端

Golang+Vue轻松构建Web应用

最近疫情在家,空闲时间比较多,整理下之前写的 Golang 项目Weave,补充了一些功能,加了前端实现。作为一个 Web 应用模板,也算是功能比较齐全了,现将开发过程中遇到的一些问题、项目特性总结下。

快速预览 Demo, 部署在Github Pages, 只提供前端功能,不包含后端与数据库。

介绍

Weave 是一个基于Go+Vue实现的 Web 应用模板,支持前后端,拥有完整的认证、存储、Restful API 等功能。

后端基于 Golang 开发,主要特性如下:

  • Restful API,通过gin实现,支持swagger
  • MVC 架构
  • 支持 Postgres 存储,可以轻松替换为 MySQL,使用gorm接入
  • Redis 缓存
  • 基于JWT认证
  • 服务优雅终止
  • 请求限速
  • Docker 容器管理,Websocket支持
  • RBAC 认证,由Casbin支持
  • 其他支持Prometheus监控、格式化日志、PProf

前端基于Vue开发,使用ElementPlus组件库

  • Vue3 开发,使用组合式 API
  • 使用vite快速编译
  • 支持WebShell,基于xtermjs
  • 图表功能,基于echarts
  • 支持WindiCSS,减少 CSS 编写

主要界面如下:

  • 登录界面 login
  • Dashboard 界面 dashboard
  • 应用界面 apps
  • WebShell 界面 webshell

项目结构

项目组织如下:

├── Dockerfile
├── Makefile
├── README.md
├── bin
├── config # server配置
├── docs # swagger 生成文件
├── document # 文档
├── go.mod
├── go.sum
├── main.go # server入口
├── pkg # server业务代码
├── scripts # 脚本
├── static # 静态文件
└── web # 前端目录

后端结构

后端按照MVC架构实现,参考了社区一些最佳实践,具体如下:

├── pkg
   ├── common # 通用包
   ├── config # 配置相关
   ├── container # 容器库
   ├── controller # 控制器层,处理HTTP请求
   ├── database # 数据库初始化,封装
   ├── metrics # 监控相关
   ├── middleware # http中间件
   ├── model # 模型层
   ├── repository # 存储层,数据持久化
   ├── server # server入口,创建router
   └── service # 逻辑层,处理业务

前端结构

前端实现Vue3实现,与一般 Vue 项目类似

web
├── README.md
├── index.html
├── node_modules
├── package-lock.json
├── package.json
├── public
   └── favicon.ico
├── src # 所有代码位于src
   ├── App.vue # Vue项目入口
   ├── assets # 静态文件
   ├── axios # http请求封装
   ├── components # Vue组件
   ├── main.js
   ├── router # 路由
   ├── utils # 工具包
   └── views # 所有页面
└── vite.config.js # vite配置

一些细节

为什么使用 JWT

主要是为了方便服务横向扩展,如果基于Cookie+SessionSession只能保存在服务端,无法进行负载均衡。另外通过 api 访问,jwt 可以放在 HTTP Header 的Bearer Token中。

当使用 Websocket 时,不支持 HTTP Header,由于认证统一在中间件中进行,可以通过简单通过cookie存储,也可以单独为 Websocket 配置认证。

JWT 不支持取消,可以通过在 redis 存入黑名单实现。

缓存实现

加入了缓存便引入了数据一致性问题,经典的解决办法是先写数据库再写缓存(Cache-Aside 模式),实现最终一致性,业务简单的项目可以使用这种方法。

那先写缓存行不行?如果同时有一个写请求一读请求,写请求会先删除缓存,读请求缓慢未命中会将 DB 中的旧数据载入,可能会造成数据不一致。先写数据库则不会有这样的问题,如果要实现先写缓存,可以使用双删的办法,即写前后分别操作一次缓存,这样处理逻辑会更复杂。如果不想侵入业务代码,可以通过监听 Binlog 来异步更新缓存。

请求限流

限流使用了golang.org/x/time/rate提供的令牌桶算法,以应对突发流量,可以对单个 IP 以及 Server 层面实现请求控制。

需要特别注意的是限流应当区别长连接与短连接,比如Weave中实现了容器exec接口,通过 Websocket 登录到容器,不应该影响其他正常请求。

从零开发前端

前端而言完全是毫无经验,选用了Vue3,主要是文档比较全面适合新手。UI 基于了ElementPlus,目前还是 Beta 版本,使用过程了也遇到了一些 Bug,生产过程中不建议用,无奈的是目前Vue3好像也没有比较成熟的 UI 库。

Vue 文档以及示例很详细,上手也挺快。主要是 CCS 不熟悉,调整样式上花了不少功夫,后来引入了WindiCSS, 只编写了少量的样式,其他全部依赖 WindiCSS 实现。其他路由、请求、图表参考对应的文档实现起来也很容易。

搭建了一个比较完整的管理平台,自己还是挺满意的,后面会不断优化,加一些其他特性。

运行

通过Docker快速预览前端, 打开https://localhost:8080

docker run -d -p 8080:80 --name weave-frontend qingwave/weave-frontend:mock

后端本地运行,需要依赖 Docker,Makefile 文件只在 Linux 下有效,其他平台请自行尝试

  1. 安装数据库 postgres 与 redis,初始化库
make init
  1. 本地运行
make run

前端使用vite编译

cd web
npm i
npm run dev

更多见ReadMe

总结

本文总结了Weave的架构与特性,以及开发过程中遇到的一些问题,从零开始实现一个完整的前后端 Web 应用,其他功能后面会不断优化。

项目链接见

Explore more in https://qingwave.github.io