最近学了学go语言,想练习一下用go开发web项目,项目结构弄个什么样呢。

去码云上面找了找,找到一个用Go语言搭建的springboot风格的web项目,拿来按自己的习惯改了改,还不错。

文末git地址 

先来看一下整体的项目结构

可以看到业务的三层结构和缓存、日志、token、全局异常等。以及一个javaer们最熟悉的application配置文件……

 

下面说一下整体逻辑

首先肯定是先来搭建一个gomod的go项目,在gomod中引入一些依赖

module go_web_test

go 1.17

require (
	github.com/allegro/bigcache v1.2.1
	github.com/gin-contrib/cors v1.3.1
	github.com/gin-gonic/gin v1.7.7
	github.com/satori/go.uuid v1.2.0
	github.com/sirupsen/logrus v1.8.1
	github.com/spf13/viper v1.9.0
	github.com/stretchr/testify v1.7.0
	gorm.io/driver/mysql v1.2.1
	gorm.io/gorm v1.22.4
)

require (
	github.com/davecgh/go-spew v1.1.1 // indirect
	github.com/fsnotify/fsnotify v1.5.1 // indirect
	github.com/gin-contrib/sse v0.1.0 // indirect
	github.com/go-playground/locales v0.13.0 // indirect
	github.com/go-playground/universal-translator v0.17.0 // indirect
	github.com/go-playground/validator/v10 v10.4.1 // indirect
	github.com/go-sql-driver/mysql v1.6.0 // indirect
	github.com/golang/protobuf v1.5.2 // indirect
	github.com/hashicorp/hcl v1.0.0 // indirect
	github.com/jinzhu/inflection v1.0.0 // indirect
	github.com/jinzhu/now v1.1.3 // indirect
	github.com/json-iterator/go v1.1.11 // indirect
	github.com/leodido/go-urn v1.2.0 // indirect
	github.com/magiconair/properties v1.8.5 // indirect
	github.com/mattn/go-isatty v0.0.12 // indirect
	github.com/mitchellh/mapstructure v1.4.2 // indirect
	github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 // indirect
	github.com/modern-go/reflect2 v1.0.1 // indirect
	github.com/pelletier/go-toml v1.9.4 // indirect
	github.com/pmezard/go-difflib v1.0.0 // indirect
	github.com/spf13/afero v1.6.0 // indirect
	github.com/spf13/cast v1.4.1 // indirect
	github.com/spf13/jwalterweatherman v1.1.0 // indirect
	github.com/spf13/pflag v1.0.5 // indirect
	github.com/subosito/gotenv v1.2.0 // indirect
	github.com/ugorji/go/codec v1.1.7 // indirect
	golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 // indirect
	golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf // indirect
	golang.org/x/text v0.3.6 // indirect
	google.golang.org/protobuf v1.27.1 // indirect
	gopkg.in/ini.v1 v1.63.2 // indirect
	gopkg.in/yaml.v2 v2.4.0 // indirect
	gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect
)

application配置文件里面自定义一些配置内容

server:
  appName: go_web_test
  port: 8888

db:
  dsn: "root:VHUKZE./start@(192.168.1.8:3306)/vhukze?charset=utf8mb4&parseTime=True&loc=Local"
  maxIdleConns: 200
  maxOpenConns: 1000
  connMaxLifetime: 60

然后看启动类,main.go

main方法启动项目,main方法中调用初始化组件的方法,把要用的组件都初始化完成。

下面还有一个自动生成表的方法,可以根据你struct定义的结构来自动生成表结构,就是没每加一个struct就要在这个方法里面加一行生成那个表的代码,在每次启动的时候也是会根据struct的字段来更新表结构的

package main

import (
	"fmt"
	logger "github.com/sirupsen/logrus"
	"go_web_test/biz/controller"
	"go_web_test/biz/dao"
	"go_web_test/config/cache"
	"go_web_test/config/db"
	"go_web_test/config/gin"
	"go_web_test/config/http"
	"go_web_test/config/log"
	"go_web_test/config/token"
	vc "go_web_test/config/viper"
	_ "net/url"
)

func main() {
	initComponents()
}

// 初始化服务所有组件
func initComponents() {
	// 初始化日志
	log.InitLogConfig()
	logger.Info("===================================================================================")
	logger.Info("Starting Application")
	// 读取本地配置文件
	vc.InitLocalConfigFile()
	// 初始化url配置
	//url.InitUrlConfig()
	// 初始化Mysql
	db.InitDbConfig()
	// 自动生成表
	autoMigrate()
	// 初始化缓存
	cache.InitBigCacheConfig()
	// 初始化Redis
	//redis.InitRedisConfig()
	// 初始化HttpClient连接池
	//http.InitHttpClientConfig()

	// 初始化token
	token.InitTokenConfig()

	// 初始化Gin
	router := gin.InitGinConfig()

	// 注册Api
	// 用户api
	controller.UserApi(router)

	// 启动Gin
	gin.RunGin(router)
}

// 自动生成表
func autoMigrate() {
	err := db.DB.AutoMigrate(dao.User{})
	if err != nil {
		_ = fmt.Errorf("自动生成user表失败")
		panic(err)
	}

}

下面来说一下初始化组件中的每个组件内容

初始化日志配置,就是配一下输出日志的格式和输入到文件什么的,下面那个LoggerAccess方法是定义了一个gin的中间件,用来输出请求的日志信息,其实格式跟gin默认的日志输出格式差不多

package log

import (
	"fmt"
	"github.com/gin-gonic/gin"
	log "github.com/sirupsen/logrus"
	"go_web_test/config/log/lumberjack"
	"go_web_test/utils"
	"io"
	"os"
	"path"
	"time"
)

func InitLogConfig() {
	// 设置日志输出路径和名称
	logFilePath := path.Join("../log/", "go_web_test.log")
	// 日志输出滚动设置
	fileOut := &lumberjack.Logger{
		Filename:   logFilePath, // 日志文件位置
		MaxSize:    100,         // 单文件最大容量,单位是MB
		MaxBackups: 500,         // 最大保留过期文件个数
		MaxAge:     15,          // 保留过期文件的最大时间间隔,单位是天
		LocalTime:  true,        // 启用当地时区计时
	}
	// 文件和控制台日志输出
	writers := []io.Writer{
		fileOut,
		os.Stdout,
	}
	fileAndStdoutWriter := io.MultiWriter(writers...)
	log.SetOutput(fileAndStdoutWriter)
	// 设置日志格式为Text格式
	log.SetFormatter(&log.TextFormatter{
		DisableColors:   false,
		FullTimestamp:   true,
		TimestampFormat: "2006-01-02 15:04:05",
	})
	// 设置日志级别为Info以上
	log.SetLevel(log.InfoLevel)
}

// LoggerAccess 入口日志打印
func LoggerAccess(c *gin.Context) {
	// 开始时间
	startTime := time.Now()
	// 处理请求
	c.Next()
	// 请求方式
	reqMethod := c.Request.Method
	// 请求路由
	reqUri := c.Request.RequestURI
	// 状态码
	statusCode := c.Writer.Status()
	// 服务器IP
	serverIP := utils.GetLocalIP()
	// 客户端IP
	clientIP := c.ClientIP()
	// 结束时间
	endTime := time.Now()
	// 执行时间
	latencyTime := fmt.Sprintf("%6v", endTime.Sub(startTime))
	//日志格式
	log.WithFields(log.Fields{
		"server-ip": serverIP,
		"duration":  latencyTime,
		"status":    statusCode,
		"method":    reqMethod,
		"uri":       reqUri,
		"client-ip": clientIP,
	}).Info("Api accessing")
}

读取本地配置文件,就是读取我们的application.yaml配置文件的内容,这里用的是viper这个工具,这里读取完之后,在其他代码里面就可以直接用viper来获取配置文件的内容了

package viper

import (
	"fmt"
	log "github.com/sirupsen/logrus"
	"github.com/spf13/viper"
)

const fileName = "application"

// InitLocalConfigFile 加载本地配置文件
func InitLocalConfigFile() {
	log.Info("初始化本地配置文件……")
	viper.SetConfigName(fileName)
	viper.SetConfigType("yaml")
	viper.AddConfigPath("./")
	err := viper.ReadInConfig()
	if err != nil {
		panic(fmt.Errorf("读取配置文件失败: %s \n", err))
	}
	log.Info("本地配置文件初始化完成……")
}

初始化mysql,根据配置文件中的URL来连接数据库

package db

import (
	"fmt"
	log "github.com/sirupsen/logrus"
	"github.com/spf13/viper"
	"gorm.io/driver/mysql"
	"gorm.io/gorm"
	"gorm.io/gorm/schema"
	"time"
)

var DB *gorm.DB

// InitDbConfig 初始化Db
func InitDbConfig() {
	log.Info("初始化数据库 Mysql")
	var err error
	dsn := viper.GetString("db.dsn")
	maxIdleConns := viper.GetInt("db.maxIdleConns")
	maxOpenConns := viper.GetInt("db.maxOpenConns")
	connMaxLifetime := viper.GetInt("db.connMaxLifetime")
	if DB, err = gorm.Open(mysql.Open(dsn), &gorm.Config{
		QueryFields: true,
		NamingStrategy: schema.NamingStrategy{
			TablePrefix:   "",   // 表名前缀
			SingularTable: true, // 使用单数表名
		},
	}); err != nil {
		panic(fmt.Errorf("初始化数据库失败: %s \n", err))
	}
	sqlDB, err := DB.DB()
	if sqlDB != nil {
		sqlDB.SetMaxIdleConns(maxIdleConns)                                    // 空闲连接数
		sqlDB.SetMaxOpenConns(maxOpenConns)                                    // 最大连接数
		sqlDB.SetConnMaxLifetime(time.Second * time.Duration(connMaxLifetime)) // 单位:秒
	}
	log.Info("Mysql: 数据库初始化完成")
}

初始化缓存,没有redis的时候可以用这个,我这里没有弄redis,先把redis的内容注释了。

package cache

import (
	"fmt"
	"github.com/allegro/bigcache"
	log "github.com/sirupsen/logrus"
	"go_web_test/utils"
	"math"
	"time"
)

var BigCache *Cache

// Cache 缓存
type Cache struct {
	BigCache *bigcache.BigCache // 本地缓存
}

// Get 根据key从缓存中获取对象
func (c Cache) Get(key string) (value interface{}, err error) {
	valueBytes, err := c.BigCache.Get(key)
	if err != nil {
		return nil, err
	}
	value = utils.Deserialize(valueBytes)
	return value, nil
}

// Set 根据key,value将目标对象存入缓存中
func (c Cache) Set(key string, value interface{}) {
	valueBytes := utils.Serialize(value)
	err := c.BigCache.Set(key, valueBytes)
	if err != nil {
		panic(err)
	}
}

// InitBigCacheConfig 初始化BigCache
func InitBigCacheConfig() {
	log.Info("初始化缓存…… BigCache")
	config := bigcache.Config{
		Shards:           1024,                      // 存储的条目数量,值必须是2的幂
		LifeWindow:       math.MaxInt16 * time.Hour, // 超时后条目被处理
		CleanWindow:      2 * time.Minute,           // 处理超时条目的时间范围
		MaxEntrySize:     500,                       // 条目最大尺寸,以字节为单位
		HardMaxCacheSize: 0,                         // 设置缓存最大值,以MB为单位,超过了不在分配内存。0表示无限制分配
	}
	bigCache, err := bigcache.NewBigCache(config)
	if err != nil {
		panic(fmt.Errorf("初始化BigCache: %s \n", err))
	}
	BigCache = &Cache{
		BigCache: bigCache,
	}
	log.Info("BigCache: 初始化完成")
}

初始化token配置,就是token验证的配置,可以配置需要忽略的请求路径

package token

var TokenCfg *TokenConfig // token配置

type TokenConfig struct {
	IgnorePaths []string
}

// AddIgnorePath 增加token不校验路径
func (config *TokenConfig) AddIgnorePath(ignorePath string) *TokenConfig {
	config.IgnorePaths = append(config.IgnorePaths, ignorePath)
	return config
}

// TokenIgnorePath token不校验路径集
func (config *TokenConfig) TokenIgnorePath() {
	config.AddIgnorePath("/token/*").
		AddIgnorePath("/ping").AddIgnorePath("/user/*")
}

// InitTokenConfig 初始化token配置
func InitTokenConfig() {
	TokenCfg = &TokenConfig{
		IgnorePaths: make([]string, 0),
	}
	TokenCfg.TokenIgnorePath()
}

 初始化gin,拿到router。可以看到这里拿到router之后use了四个中间件,日志中间件、全局异常处理中间件,token验证中间件,跨域处理中间件。跨域中间件是用的gin相关库里面的,token验证中间件在后面。然后在RunGin方法中用指定的端口启动了gin

package gin

import (
	"fmt"
	"github.com/gin-contrib/cors"
	"github.com/gin-gonic/gin"
	logger "github.com/sirupsen/logrus"
	"github.com/spf13/viper"
	"go_web_test/config/log"
	"go_web_test/config/token"
	err "go_web_test/exception"
)

// InitGinConfig 初始化Gin
func InitGinConfig() *gin.Engine {
	logger.Info("初始化 gin……")
	gin.SetMode(gin.ReleaseMode)
	router := gin.Default()
	// 入口日志打印
	router.Use(log.LoggerAccess)
	// 统一异常处理
	router.Use(err.ErrHandle)
	// 跨域处理
	router.Use(cors.Default())
	// token校验
	router.Use(token.TokenVerify)
	// 健康检测
	router.GET("/ping", func(c *gin.Context) {
		c.JSON(200, gin.H{
			"message": "pong",
		})
	})
	logger.Info("Gin: 初始化完成……")
	return router
}

// RunGin 启动Gin
func RunGin(router *gin.Engine) {
	port := viper.GetString("server.port")
	logger.Info(fmt.Sprintf("Service started on port(s): %s", port))
	_ = router.Run(":" + port)
}

token验证的中间件,先判断一下当前请求路径需不需要验证,验证失败就抛出一个token验证失败的异常,这个时候全局异常处理里面就可以捕获到处理并返回错误信息

package token

import (
	"github.com/gin-gonic/gin"
	"go_web_test/biz/dto"
	"go_web_test/config/cache"
	"strings"
)

func TokenVerify(c *gin.Context) {
	request := c.Request
	// 过滤不用token校验的url
	if noTokenVerify(TokenCfg.IgnorePaths, request.RequestURI) {
		return
	}

	// 获取token
	tokenStr := request.Header.Get("token")
	if len(tokenStr) == 0 {
		panic(NewTokenError(dto.Unauthorized, dto.GetResultMsg(dto.Unauthorized)))
	}

	if _, err := cache.BigCache.Get(tokenStr); err != nil {
		panic(NewTokenError(dto.Unauthorized, dto.GetResultMsg(dto.Unauthorized)))
	}

	c.Next()
}

// noTokenVerify 判断url是否不需要token校验
func noTokenVerify(ignorePaths []string, path string) bool {
	// 查询缓存
	if noVerify, err := cache.BigCache.Get(path); err == nil {
		return noVerify.(bool)
	}
	// 匹配url
	for _, ignorePath := range ignorePaths {
		// 路径尾通配符*过滤
		if strings.LastIndex(ignorePath, "*") == len(ignorePath)-1 {
			ignorePath = strings.Split(ignorePath, "*")[0]
			if endIndex := strings.LastIndex(path, "/"); strings.Compare(path[0:endIndex+1], ignorePath) == 0 {
				// 添加缓存
				cache.BigCache.Set(path, true)
				return true
			}
			// 无通配符*过滤
		} else if strings.Compare(path, ignorePath) == 0 {
			// 添加缓存
			cache.BigCache.Set(path, true)
			return true
		}
	}
	return false
}

全局异常

package exception

import (
	"github.com/gin-gonic/gin"
	log "github.com/sirupsen/logrus"
	"go_web_test/biz/dto"
	"go_web_test/config/token"
	"net/http"
	"runtime/debug"
)

// ErrHandle 统一异常处理
func ErrHandle(c *gin.Context) {
	defer func() {
		if r := recover(); r != nil {
			apiErr, isApiErr := r.(*ApiError)
			tokenErr, isTokenErr := r.(*token.TokenError)
			if isApiErr {
				// 打印错误堆栈信息
				log.WithField("ErrMsg", apiErr.Error()).Error("PanicHandler handled apiError: ")
				// 封装通用json返回
				c.JSON(http.StatusInternalServerError, apiErr)
			} else if isTokenErr {
				// 打印错误堆栈信息
				log.WithField("ErrMsg", tokenErr.Error()).Error("PanicHandler handled tokenError: ")
				// 封装通用json返回
				c.JSON(http.StatusUnauthorized, tokenErr)
			} else {
				// 打印错误堆栈信息
				err := r.(error)
				log.WithField("ErrMsg", err.Error()).Error("PanicHandler handled ordinaryError: ")
				debug.PrintStack()
				// 封装通用json返回
				c.JSON(http.StatusInternalServerError, NewApiError(dto.InternalServerError, dto.GetResultMsg(dto.InternalServerError)))
			}
			c.Abort()
		}
	}()
	c.Next()
}

最后就是注册api了,每写一个controller就要在这里注册一下,然后启动gin。

说了一堆配置相关的,来看一下业务的三层结构吧

controller层

使用struct定义一个UserHandler,在注册api的方法中给service赋值,这里可以看到service也是有一个接口和实现类的

package controller

import (
	"github.com/gin-gonic/gin"
	"go_web_test/biz/dto"
	"go_web_test/biz/service"
	"net/http"
	"strconv"
)

type UserHandler struct {
	userService service.UserService
}

func UserApi(router *gin.Engine) {

	userHandler := UserHandler{
		userService: &service.UserServiceImpl{},
	}

	userGroup := router.Group("user/")
	{
		userGroup.GET("/:id", userHandler.user)
	}
}

// 根据ID查询用户
func (userHandler UserHandler) user(c *gin.Context) {
	userIdStr := c.Param("id")
	userId, _ := strconv.Atoi(userIdStr)
	user := userHandler.userService.User(userId)

	c.JSON(http.StatusOK, dto.Ok(user))
}

service层

先定义一个接口,然后一个实现,go语言中的实现没有关键字,遵循duck-typeing的原则,只要像,那就是它的实现。在idea中是会有左边的跳转上下箭头的。同样在service里面也有一个dao层的引用

package service

import "go_web_test/biz/dao"

type UserService interface {
	User(userId int) *dao.User
}

type UserServiceImpl struct {
}

func (UserServiceImpl) User(userId int) *dao.User {
	user := &dao.User{}
	user.SelectById(userId)
	return user
}

dao层

这个里面有对应数据库表的结构以及这个结构所属的方法

package dao

import "go_web_test/config/db"

type User struct {
	Id   int    `json:"id" gorm:"primary_key"`
	Name string `json:"name" gorm:"size:50"`
}

func (user *User) SelectById(userId int) {
	db.DB.First(&user, userId)
}

有一点要注意的是,这里在设计的时候要避免循环依赖问题,毕竟没有spring来帮我们解决循环依赖了。其实循环依赖这种问题本来就是要在设计上避免,而不是代码中去解决它吧