repository.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  1. package repository
  2. import (
  3. "context"
  4. "fmt"
  5. "github.com/casbin/casbin/v2"
  6. "github.com/casbin/casbin/v2/model"
  7. gormadapter "github.com/casbin/gorm-adapter/v3"
  8. "github.com/glebarez/sqlite"
  9. "github.com/go-nunu/nunu-layout-advanced/pkg/log"
  10. "github.com/go-nunu/nunu-layout-advanced/pkg/rabbitmq"
  11. "github.com/go-nunu/nunu-layout-advanced/pkg/sharding"
  12. "github.com/go-nunu/nunu-layout-advanced/pkg/zapgorm2"
  13. "github.com/qiniu/qmgo"
  14. "github.com/redis/go-redis/v9"
  15. "github.com/spf13/viper"
  16. "gorm.io/driver/mysql"
  17. "gorm.io/driver/postgres"
  18. "gorm.io/gorm"
  19. gormlogger "gorm.io/gorm/logger"
  20. "gorm.io/plugin/dbresolver"
  21. "time"
  22. )
  23. const ctxTxKey = "TxKey"
  24. type Repository struct {
  25. Db *gorm.DB
  26. Rdb *redis.Client
  27. mongoClient *qmgo.Client
  28. MongoDB *qmgo.Database
  29. mq *rabbitmq.RabbitMQ
  30. Logger *log.Logger
  31. E *casbin.SyncedEnforcer
  32. Manager *sharding.ShardingManager
  33. }
  34. func NewRepository(
  35. logger *log.Logger,
  36. db *gorm.DB,
  37. rdb *redis.Client,
  38. mongoClient *qmgo.Client,
  39. mongoDB *qmgo.Database,
  40. mq *rabbitmq.RabbitMQ,
  41. e *casbin.SyncedEnforcer,
  42. manager *sharding.ShardingManager,
  43. ) *Repository {
  44. return &Repository{
  45. Db: db,
  46. Rdb: rdb,
  47. mongoClient: mongoClient,
  48. MongoDB: mongoDB,
  49. mq: mq,
  50. Logger: logger,
  51. E: e,
  52. Manager: manager,
  53. }
  54. }
  55. type Transaction interface {
  56. Transaction(ctx context.Context, fn func(ctx context.Context) error) error
  57. // 在特定数据库上执行事务
  58. TransactionWithDB(ctx context.Context, dbName string, fn func(ctx context.Context) error) error
  59. }
  60. func NewTransaction(r *Repository) Transaction {
  61. return r
  62. }
  63. // DB return tx
  64. // If you need to create a Transaction, you must call DB(ctx) and Transaction(ctx,fn)
  65. func (r *Repository) DB(ctx context.Context) *gorm.DB {
  66. v := ctx.Value(ctxTxKey)
  67. if v != nil {
  68. if tx, ok := v.(*gorm.DB); ok {
  69. return tx
  70. }
  71. }
  72. return r.Db.WithContext(ctx)
  73. }
  74. // DBWithName 使用特定名称的数据库连接
  75. func (r *Repository) DBWithName(ctx context.Context, dbName string) *gorm.DB {
  76. // 先检查上下文中是否已存在事务
  77. v := ctx.Value(ctxTxKey)
  78. if v != nil {
  79. if tx, ok := v.(*gorm.DB); ok {
  80. // 如果事务中已经指定了数据库,则直接返回
  81. return tx
  82. }
  83. }
  84. // 使用指定名称的数据库连接
  85. if dbName != "" {
  86. return r.Db.Clauses(dbresolver.Use(dbName)).WithContext(ctx)
  87. }
  88. return r.Db.WithContext(ctx)
  89. }
  90. func (r *Repository) Transaction(ctx context.Context, fn func(ctx context.Context) error) error {
  91. return r.Db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
  92. ctxWithTx := context.WithValue(ctx, ctxTxKey, tx)
  93. return fn(ctxWithTx)
  94. })
  95. }
  96. // TransactionWithDB 在特定数据库上执行事务
  97. func (r *Repository) TransactionWithDB(ctx context.Context, dbName string, fn func(ctx context.Context) error) error {
  98. // 使用特定的数据库连接
  99. db := r.Db
  100. if dbName != "" {
  101. db = db.Clauses(dbresolver.Use(dbName))
  102. }
  103. return db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
  104. // tx已经是针对特定数据库的事务句柄,无需再次指定数据库
  105. ctxWithTx := context.WithValue(ctx, ctxTxKey, tx)
  106. return fn(ctxWithTx)
  107. })
  108. }
  109. func NewDB(conf *viper.Viper, l *log.Logger) *gorm.DB {
  110. var (
  111. db *gorm.DB
  112. err error
  113. )
  114. // 获取主数据库键名
  115. primaryDBKey := conf.GetString("data.primary_db_key")
  116. if primaryDBKey == "" {
  117. // 默认使用user作为主数据库键名(向后兼容)
  118. primaryDBKey = "user"
  119. }
  120. // 从配置中获取主数据库配置
  121. driver := conf.GetString(fmt.Sprintf("data.db.%s.driver", primaryDBKey))
  122. if driver == "" {
  123. panic("主数据库驱动配置不能为空")
  124. }
  125. dsn := conf.GetString(fmt.Sprintf("data.db.%s.dsn", primaryDBKey))
  126. if dsn == "" {
  127. panic("主数据库连接字符串不能为空")
  128. }
  129. // 读取日志级别配置
  130. logLevelStr := conf.GetString(fmt.Sprintf("data.db.%s.logLevel", primaryDBKey))
  131. var logLevel gormlogger.LogLevel
  132. switch logLevelStr {
  133. case "silent":
  134. logLevel = gormlogger.Silent
  135. case "error":
  136. logLevel = gormlogger.Error
  137. case "warn":
  138. logLevel = gormlogger.Warn
  139. case "info":
  140. logLevel = gormlogger.Info
  141. default:
  142. // MySQL 默认只记录警告和错误
  143. if driver == "mysql" {
  144. logLevel = gormlogger.Warn
  145. } else {
  146. logLevel = gormlogger.Info
  147. }
  148. }
  149. logger := zapgorm2.New(l.Logger).LogMode(logLevel)
  150. // 连接主数据库
  151. switch driver {
  152. case "mysql":
  153. db, err = gorm.Open(mysql.Open(dsn), &gorm.Config{
  154. Logger: logger,
  155. })
  156. case "postgres":
  157. db, err = gorm.Open(postgres.New(postgres.Config{
  158. DSN: dsn,
  159. PreferSimpleProtocol: true,
  160. }), &gorm.Config{
  161. Logger: logger,
  162. })
  163. case "sqlite":
  164. db, err = gorm.Open(sqlite.Open(dsn), &gorm.Config{
  165. Logger: logger,
  166. })
  167. default:
  168. panic("不支持的数据库驱动类型: " + driver)
  169. }
  170. if err != nil {
  171. panic(fmt.Sprintf("连接主数据库失败: %s", err.Error()))
  172. }
  173. // 创建 dbresolver 实例
  174. resolver := dbresolver.Register(dbresolver.Config{})
  175. // 获取所有配置的数据库列表
  176. databases := conf.GetStringMap("data.db")
  177. // 遍历所有数据库配置(跳过主数据库,因为已经连接)
  178. for dbKey, _ := range databases {
  179. // 跳过主数据库(已经直接连接了)
  180. if dbKey == primaryDBKey {
  181. continue
  182. }
  183. // 检查该键是否确实是一个数据库配置对象
  184. dbDriver := conf.GetString(fmt.Sprintf("data.db.%s.driver", dbKey))
  185. dbDSN := conf.GetString(fmt.Sprintf("data.db.%s.dsn", dbKey))
  186. if dbDriver != "" && dbDSN != "" {
  187. // 构建数据库连接器
  188. var dialector gorm.Dialector
  189. switch dbDriver {
  190. case "mysql":
  191. dialector = mysql.Open(dbDSN)
  192. case "postgres":
  193. dialector = postgres.New(postgres.Config{
  194. DSN: dbDSN,
  195. PreferSimpleProtocol: true,
  196. })
  197. case "sqlite":
  198. dialector = sqlite.Open(dbDSN)
  199. default:
  200. l.Warn(fmt.Sprintf("跳过不支持的数据库驱动类型: %s (dbKey: %s)", dbDriver, dbKey))
  201. continue
  202. }
  203. // 注册到resolver
  204. resolver.Register(dbresolver.Config{
  205. Sources: []gorm.Dialector{dialector},
  206. Replicas: []gorm.Dialector{dialector},
  207. Policy: dbresolver.RandomPolicy{},
  208. }, dbKey) // 使用配置键作为数据库名称
  209. l.Info(fmt.Sprintf("成功配置数据库连接: %s", dbKey))
  210. }
  211. }
  212. // 设置连接池参数
  213. resolver.SetConnMaxIdleTime(time.Hour).
  214. SetConnMaxLifetime(24 * time.Hour).
  215. SetMaxIdleConns(10).
  216. SetMaxOpenConns(100)
  217. // 应用配置好的 dbresolver 到 db
  218. err = db.Use(resolver)
  219. if err != nil {
  220. panic(fmt.Sprintf("应用数据库连接配置失败: %s", err.Error()))
  221. }
  222. // 主数据库连接池配置
  223. sqlDB, err := db.DB()
  224. if err != nil {
  225. panic(err)
  226. }
  227. sqlDB.SetMaxIdleConns(10)
  228. sqlDB.SetMaxOpenConns(100)
  229. sqlDB.SetConnMaxLifetime(time.Hour)
  230. return db
  231. }
  232. func NewRedis(conf *viper.Viper) *redis.Client {
  233. rdb := redis.NewClient(&redis.Options{
  234. Addr: conf.GetString("data.redis.addr"),
  235. Password: conf.GetString("data.redis.password"),
  236. DB: conf.GetInt("data.redis.db"),
  237. })
  238. ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
  239. defer cancel()
  240. _, err := rdb.Ping(ctx).Result()
  241. if err != nil {
  242. panic(fmt.Sprintf("redis error: %s", err.Error()))
  243. }
  244. return rdb
  245. }
  246. func NewMongoClient(conf *viper.Viper) *qmgo.Client {
  247. timeout := conf.GetDuration("data.mongodb.timeout")
  248. if timeout == 0 {
  249. timeout = 10 * time.Second
  250. }
  251. maxPoolSize := conf.GetUint64("data.mongodb.max_pool_size")
  252. ctx, cancel := context.WithTimeout(context.Background(), timeout)
  253. defer cancel()
  254. // 创建连接配置
  255. clientOpts := &qmgo.Config{
  256. Uri: conf.GetString("data.mongodb.uri"),
  257. MaxPoolSize: &maxPoolSize,
  258. }
  259. // 连接到MongoDB
  260. client, err := qmgo.NewClient(ctx, clientOpts)
  261. if err != nil {
  262. panic(fmt.Sprintf("连接MongoDB失败: %s", err.Error()))
  263. }
  264. return client
  265. }
  266. func NewMongoDB(client *qmgo.Client, conf *viper.Viper) *qmgo.Database {
  267. databaseName := conf.GetString("data.mongodb.database")
  268. if databaseName == "" {
  269. panic("MongoDB数据库名不能为空")
  270. }
  271. return client.Database(databaseName)
  272. }
  273. func NewRabbitMQ(conf *viper.Viper, logger *log.Logger) (*rabbitmq.RabbitMQ, func()) {
  274. var cfg rabbitmq.Config
  275. if err := conf.UnmarshalKey("rabbitmq", &cfg); err != nil {
  276. panic(fmt.Sprintf("unmarshal rabbitmq config error: %s", err.Error()))
  277. }
  278. mq, err := rabbitmq.New(cfg, logger)
  279. if err != nil {
  280. panic(fmt.Sprintf("init rabbitmq error: %s", err.Error()))
  281. }
  282. // Setup task queue
  283. if err := mq.SetupAllTaskQueues(); err != nil {
  284. panic(fmt.Sprintf("failed to setup rabbitmq task queues: %v", err))
  285. }
  286. cleanup := func() {
  287. logger.Info("Closing RabbitMQ connection")
  288. _ = mq.Close()
  289. }
  290. return mq, cleanup
  291. }
  292. func NewCasbinEnforcer(conf *viper.Viper, l *log.Logger, db *gorm.DB) *casbin.SyncedEnforcer {
  293. var (
  294. adapter *gormadapter.Adapter
  295. err error
  296. casbinDb *gorm.DB = db // 默认使用主数据库连接
  297. )
  298. // 创建一个专门给Enforcer使用的、日志级别为Warn的日志记录器,以屏蔽轮询日志。
  299. // 这不会影响数据库连接的全局日志配置。
  300. enforcerLogger := zapgorm2.New(l.Logger).LogMode(gormlogger.Warn)
  301. // 扫描配置,查找为Casbin指定的数据库
  302. dbSettings := conf.GetStringMap("data.db")
  303. foundSpecialDb := false
  304. for dbKey := range dbSettings {
  305. casbinFlagPath := fmt.Sprintf("data.db.%s.casbin", dbKey)
  306. if conf.GetBool(casbinFlagPath) {
  307. l.Info(fmt.Sprintf("检测到Casbin专用数据库配置: '%s'。Enforcer将使用此数据库连接。", dbKey))
  308. // 从全局连接池中获取指定的数据库连接
  309. casbinDb = db.Clauses(dbresolver.Use(dbKey))
  310. foundSpecialDb = true
  311. break
  312. }
  313. }
  314. if !foundSpecialDb {
  315. l.Warn("未找到Casbin的专用数据库配置 (缺少 'casbin: true' 标志),Enforcer将回退使用主数据库连接。")
  316. }
  317. // 为Enforcer创建一个带有“安静”日志记录器的GORM会话。
  318. // 这样可以确保只有Enforcer自身的操作是安静的,
  319. // 而通过DBWithName()进行的直接数据库操作仍将使用原始的、更详细的日志记录器。
  320. dbForEnforcer := casbinDb.Session(&gorm.Session{Logger: enforcerLogger})
  321. // 使用带有“安静”日志记录器的会话来创建适配器
  322. adapter, err = gormadapter.NewAdapterByDB(dbForEnforcer)
  323. if err != nil {
  324. panic(fmt.Sprintf("创建Casbin gorm-adapter失败: %s", err))
  325. }
  326. m, err := model.NewModelFromString(`
  327. [request_definition]
  328. r = sub, obj, act
  329. [policy_definition]
  330. p = sub, obj, act
  331. [role_definition]
  332. g = _, _
  333. [policy_effect]
  334. e = some(where (p.eft == allow))
  335. [matchers]
  336. m = g(r.sub, p.sub) && r.obj == p.obj && r.act == p.act
  337. `)
  338. if err != nil {
  339. panic(fmt.Sprintf("创建Casbin模型失败: %s", err))
  340. }
  341. e, err := casbin.NewSyncedEnforcer(m, adapter)
  342. if err != nil {
  343. panic(fmt.Sprintf("创建Casbin Enforcer失败: %s", err))
  344. }
  345. // 每10秒自动加载策略
  346. e.StartAutoLoadPolicy(10 * time.Second)
  347. // 自动保存策略
  348. e.EnableAutoSave(true)
  349. return e
  350. }
  351. // NewShardingManager creates a ShardingManager with threshold support for dependency injection
  352. func NewShardingManager(logger *log.Logger) *sharding.ShardingManager {
  353. strategy := sharding.NewMonthlyShardingStrategy()
  354. // 配置阈值参数 - 统一管理所有表的阈值
  355. thresholdConfig := &sharding.ThresholdConfig{
  356. Enabled: true,
  357. MaxRows: 3000000, // 默认300万条
  358. TableThresholds: map[string]int64{
  359. "log": 3000000, // log表300万条
  360. "waf_log": 5000000, // waf_log表500万条
  361. },
  362. }
  363. return sharding.NewShardingManagerWithThreshold(strategy, logger, thresholdConfig)
  364. }