370 lines
8.4 KiB
Go
370 lines
8.4 KiB
Go
package data
|
||
|
||
import (
|
||
"context"
|
||
"fmt"
|
||
"time"
|
||
|
||
"kra/internal/biz/example"
|
||
"kra/internal/conf"
|
||
dataexample "kra/internal/data/example"
|
||
datasystem "kra/internal/data/system"
|
||
pkgcasbin "kra/pkg/casbin"
|
||
|
||
"github.com/casbin/casbin/v2"
|
||
oracle "github.com/dzwvip/gorm-oracle"
|
||
"github.com/glebarez/sqlite"
|
||
_ "github.com/go-sql-driver/mysql"
|
||
"github.com/google/wire"
|
||
"github.com/redis/go-redis/v9"
|
||
"gorm.io/driver/mysql"
|
||
"gorm.io/driver/postgres"
|
||
"gorm.io/driver/sqlserver"
|
||
"gorm.io/gorm"
|
||
"gorm.io/gorm/logger"
|
||
"gorm.io/gorm/schema"
|
||
)
|
||
|
||
// ProviderSet is data providers.
|
||
var ProviderSet = wire.NewSet(
|
||
NewDB,
|
||
NewData,
|
||
NewRedisClient,
|
||
NewCasbinEnforcer,
|
||
NewOssUploader,
|
||
NewDataAuthorityProvider,
|
||
// System
|
||
datasystem.NewUserRepo,
|
||
datasystem.NewApiRepo,
|
||
datasystem.NewAuthorityRepo,
|
||
datasystem.NewAuthorityBtnRepo,
|
||
datasystem.NewCasbinRepo,
|
||
datasystem.NewMenuRepo,
|
||
datasystem.NewDictionaryRepo,
|
||
datasystem.NewDictionaryDetailRepo,
|
||
datasystem.NewJwtBlacklistRepo,
|
||
datasystem.NewOperationRecordRepo,
|
||
datasystem.NewParamsRepo,
|
||
datasystem.NewErrorRepo,
|
||
datasystem.NewVersionRepo,
|
||
datasystem.NewExportTemplateRepo,
|
||
datasystem.NewAutoCodeRepo,
|
||
datasystem.NewAutoCodeHistoryRepo,
|
||
datasystem.NewAutoCodePackageRepo,
|
||
datasystem.NewAutoCodeTemplateRepo,
|
||
datasystem.NewAutoCodePluginRepo,
|
||
// Example
|
||
dataexample.NewFileUploadRepo,
|
||
dataexample.NewCustomerRepo,
|
||
dataexample.NewAttachmentCategoryRepo,
|
||
dataexample.NewBreakpointContinueRepo,
|
||
)
|
||
|
||
// Data 数据层包装器
|
||
type Data struct {
|
||
db *gorm.DB
|
||
redis redis.UniversalClient
|
||
}
|
||
|
||
// NewData 创建数据层包装器
|
||
func NewData(db *gorm.DB, redisClient redis.UniversalClient) (*Data, func(), error) {
|
||
cleanup := func() {
|
||
// 关闭数据库连接
|
||
if db != nil {
|
||
sqlDB, err := db.DB()
|
||
if err == nil {
|
||
sqlDB.Close()
|
||
}
|
||
}
|
||
// 关闭Redis连接
|
||
if redisClient != nil {
|
||
redisClient.Close()
|
||
}
|
||
}
|
||
return &Data{db: db, redis: redisClient}, cleanup, nil
|
||
}
|
||
|
||
// DB 获取数据库连接
|
||
func (d *Data) DB() *gorm.DB {
|
||
return d.db
|
||
}
|
||
|
||
// Redis 获取Redis客户端
|
||
func (d *Data) Redis() redis.UniversalClient {
|
||
return d.redis
|
||
}
|
||
|
||
// NewDB 创建数据库连接(根据配置的数据库类型)
|
||
func NewDB(system *conf.System, mysqlCfg *conf.Mysql, pgsqlCfg *conf.Pgsql, sqliteCfg *conf.Sqlite, mssqlCfg *conf.Mssql, oracleCfg *conf.Oracle) (*gorm.DB, error) {
|
||
dbType := "mysql"
|
||
if system != nil && system.DbType != "" {
|
||
dbType = system.DbType
|
||
}
|
||
|
||
switch dbType {
|
||
case "mysql":
|
||
return NewMysqlDB(mysqlCfg)
|
||
case "pgsql":
|
||
return NewPgsqlDB(pgsqlCfg)
|
||
case "sqlite":
|
||
return NewSqliteDB(sqliteCfg)
|
||
case "mssql":
|
||
return NewMssqlDB(mssqlCfg)
|
||
case "oracle":
|
||
return NewOracleDB(oracleCfg)
|
||
default:
|
||
return NewMysqlDB(mysqlCfg)
|
||
}
|
||
}
|
||
|
||
// NewMysqlDB 创建MySQL数据库连接
|
||
func NewMysqlDB(c *conf.Mysql) (*gorm.DB, error) {
|
||
if c == nil || c.DbName == "" {
|
||
return nil, fmt.Errorf("mysql config is empty")
|
||
}
|
||
|
||
dsn := fmt.Sprintf("%s:%s@tcp(%s:%s)/%s?%s",
|
||
c.Username, c.Password, c.Path, c.Port, c.DbName, c.Config)
|
||
|
||
gormConfig := newGormConfig(c.Prefix, c.Singular, c.LogMode)
|
||
|
||
db, err := gorm.Open(mysql.New(mysql.Config{
|
||
DSN: dsn,
|
||
DefaultStringSize: 191,
|
||
SkipInitializeWithVersion: false,
|
||
}), gormConfig)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
if c.Engine != "" {
|
||
db.InstanceSet("gorm:table_options", "ENGINE="+c.Engine)
|
||
}
|
||
|
||
sqlDB, err := db.DB()
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB.SetMaxIdleConns(int(c.MaxIdleConns))
|
||
sqlDB.SetMaxOpenConns(int(c.MaxOpenConns))
|
||
|
||
return db, nil
|
||
}
|
||
|
||
// NewPgsqlDB 创建PostgreSQL数据库连接
|
||
func NewPgsqlDB(c *conf.Pgsql) (*gorm.DB, error) {
|
||
if c == nil || c.DbName == "" {
|
||
return nil, fmt.Errorf("pgsql config is empty")
|
||
}
|
||
|
||
dsn := fmt.Sprintf("host=%s user=%s password=%s dbname=%s port=%s %s",
|
||
c.Path, c.Username, c.Password, c.DbName, c.Port, c.Config)
|
||
|
||
gormConfig := newGormConfig(c.Prefix, c.Singular, c.LogMode)
|
||
|
||
db, err := gorm.Open(postgres.New(postgres.Config{
|
||
DSN: dsn,
|
||
PreferSimpleProtocol: false,
|
||
}), gormConfig)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB, err := db.DB()
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB.SetMaxIdleConns(int(c.MaxIdleConns))
|
||
sqlDB.SetMaxOpenConns(int(c.MaxOpenConns))
|
||
|
||
return db, nil
|
||
}
|
||
|
||
// NewSqliteDB 创建SQLite数据库连接
|
||
func NewSqliteDB(c *conf.Sqlite) (*gorm.DB, error) {
|
||
if c == nil || c.DbName == "" {
|
||
return nil, fmt.Errorf("sqlite config is empty")
|
||
}
|
||
|
||
dsn := c.DbName
|
||
if c.Path != "" {
|
||
dsn = fmt.Sprintf("%s/%s", c.Path, c.DbName)
|
||
}
|
||
|
||
gormConfig := newGormConfig(c.Prefix, c.Singular, c.LogMode)
|
||
|
||
db, err := gorm.Open(sqlite.Open(dsn), gormConfig)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB, err := db.DB()
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB.SetMaxIdleConns(int(c.MaxIdleConns))
|
||
sqlDB.SetMaxOpenConns(int(c.MaxOpenConns))
|
||
|
||
return db, nil
|
||
}
|
||
|
||
// NewMssqlDB 创建MSSQL数据库连接
|
||
func NewMssqlDB(c *conf.Mssql) (*gorm.DB, error) {
|
||
if c == nil || c.DbName == "" {
|
||
return nil, fmt.Errorf("mssql config is empty")
|
||
}
|
||
|
||
dsn := fmt.Sprintf("sqlserver://%s:%s@%s:%s?database=%s&%s",
|
||
c.Username, c.Password, c.Path, c.Port, c.DbName, c.Config)
|
||
|
||
gormConfig := newGormConfig(c.Prefix, c.Singular, c.LogMode)
|
||
|
||
db, err := gorm.Open(sqlserver.New(sqlserver.Config{
|
||
DSN: dsn,
|
||
DefaultStringSize: 191,
|
||
}), gormConfig)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
if c.Engine != "" {
|
||
db.InstanceSet("gorm:table_options", "ENGINE="+c.Engine)
|
||
}
|
||
|
||
sqlDB, err := db.DB()
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB.SetMaxIdleConns(int(c.MaxIdleConns))
|
||
sqlDB.SetMaxOpenConns(int(c.MaxOpenConns))
|
||
|
||
return db, nil
|
||
}
|
||
|
||
// NewOracleDB 创建Oracle数据库连接
|
||
func NewOracleDB(c *conf.Oracle) (*gorm.DB, error) {
|
||
if c == nil || c.DbName == "" {
|
||
return nil, fmt.Errorf("oracle config is empty")
|
||
}
|
||
|
||
dsn := fmt.Sprintf("oracle://%s:%s@%s:%s/%s?%s",
|
||
c.Username, c.Password, c.Path, c.Port, c.DbName, c.Config)
|
||
|
||
gormConfig := newGormConfig(c.Prefix, c.Singular, c.LogMode)
|
||
|
||
db, err := gorm.Open(oracle.Open(dsn), gormConfig)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB, err := db.DB()
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
sqlDB.SetMaxIdleConns(int(c.MaxIdleConns))
|
||
sqlDB.SetMaxOpenConns(int(c.MaxOpenConns))
|
||
|
||
return db, nil
|
||
}
|
||
|
||
// newGormConfig 创建GORM配置
|
||
func newGormConfig(prefix string, singular bool, logMode string) *gorm.Config {
|
||
return &gorm.Config{
|
||
Logger: logger.New(
|
||
&gormLogWriter{},
|
||
logger.Config{
|
||
SlowThreshold: 200 * time.Millisecond,
|
||
LogLevel: parseLogLevel(logMode),
|
||
Colorful: true,
|
||
},
|
||
),
|
||
NamingStrategy: schema.NamingStrategy{
|
||
TablePrefix: prefix,
|
||
SingularTable: singular,
|
||
},
|
||
DisableForeignKeyConstraintWhenMigrating: true,
|
||
}
|
||
}
|
||
|
||
// gormLogWriter GORM日志写入器
|
||
type gormLogWriter struct{}
|
||
|
||
func (w *gormLogWriter) Printf(format string, args ...interface{}) {
|
||
fmt.Printf(format+"\n", args...)
|
||
}
|
||
|
||
// parseLogLevel 解析日志级别
|
||
func parseLogLevel(level string) logger.LogLevel {
|
||
switch level {
|
||
case "silent":
|
||
return logger.Silent
|
||
case "error":
|
||
return logger.Error
|
||
case "warn":
|
||
return logger.Warn
|
||
case "info":
|
||
return logger.Info
|
||
default:
|
||
return logger.Info
|
||
}
|
||
}
|
||
|
||
// NewRedisClient 创建Redis客户端
|
||
func NewRedisClient(c *conf.Redis, system *conf.System) (redis.UniversalClient, error) {
|
||
// 如果不使用Redis,返回nil
|
||
if system != nil && !system.UseRedis {
|
||
return nil, nil
|
||
}
|
||
|
||
if c == nil {
|
||
return nil, nil
|
||
}
|
||
|
||
var client redis.UniversalClient
|
||
|
||
// 使用集群模式
|
||
if c.UseCluster {
|
||
client = redis.NewClusterClient(&redis.ClusterOptions{
|
||
Addrs: c.ClusterAddrs,
|
||
Password: c.Password,
|
||
})
|
||
} else {
|
||
// 使用单例模式
|
||
client = redis.NewClient(&redis.Options{
|
||
Addr: c.Addr,
|
||
Password: c.Password,
|
||
DB: int(c.Db),
|
||
})
|
||
}
|
||
|
||
// 测试连接
|
||
_, err := client.Ping(context.Background()).Result()
|
||
if err != nil {
|
||
return nil, fmt.Errorf("redis connect ping failed: %v", err)
|
||
}
|
||
|
||
return client, nil
|
||
}
|
||
|
||
// NewCasbinEnforcer 创建Casbin Enforcer
|
||
func NewCasbinEnforcer(db *gorm.DB) (*casbin.SyncedCachedEnforcer, error) {
|
||
return pkgcasbin.InitCasbin(db, "")
|
||
}
|
||
|
||
// NewOssUploader 创建OSS上传器(默认使用空实现)
|
||
func NewOssUploader() example.OssUploader {
|
||
// TODO: 根据配置返回实际的OSS实现
|
||
return &example.NilOssUploader{}
|
||
}
|
||
|
||
// NewDataAuthorityProvider 创建数据权限提供者(默认使用简单实现)
|
||
func NewDataAuthorityProvider() example.DataAuthorityProvider {
|
||
// TODO: 根据需要返回实际的数据权限实现
|
||
return &example.SimpleDataAuthorityProvider{}
|
||
}
|