123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121 |
- package repository
- import (
- "context"
- "fmt"
- "github.com/glebarez/sqlite"
- "github.com/go-nunu/nunu-layout-advanced/pkg/log"
- "github.com/go-nunu/nunu-layout-advanced/pkg/zapgorm2"
- "github.com/redis/go-redis/v9"
- "github.com/spf13/viper"
- "gorm.io/driver/mysql"
- "gorm.io/driver/postgres"
- "gorm.io/gorm"
- "time"
- )
- const ctxTxKey = "TxKey"
- type Repository struct {
- db *gorm.DB
- //rdb *redis.Client
- logger *log.Logger
- }
- func NewRepository(
- logger *log.Logger,
- db *gorm.DB,
- // rdb *redis.Client,
- ) *Repository {
- return &Repository{
- db: db,
- //rdb: rdb,
- logger: logger,
- }
- }
- type Transaction interface {
- Transaction(ctx context.Context, fn func(ctx context.Context) error) error
- }
- func NewTransaction(r *Repository) Transaction {
- return r
- }
- // DB return tx
- // If you need to create a Transaction, you must call DB(ctx) and Transaction(ctx,fn)
- func (r *Repository) DB(ctx context.Context) *gorm.DB {
- v := ctx.Value(ctxTxKey)
- if v != nil {
- if tx, ok := v.(*gorm.DB); ok {
- return tx
- }
- }
- return r.db.WithContext(ctx)
- }
- func (r *Repository) Transaction(ctx context.Context, fn func(ctx context.Context) error) error {
- return r.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
- ctx = context.WithValue(ctx, ctxTxKey, tx)
- return fn(ctx)
- })
- }
- func NewDB(conf *viper.Viper, l *log.Logger) *gorm.DB {
- var (
- db *gorm.DB
- err error
- )
- logger := zapgorm2.New(l.Logger)
- driver := conf.GetString("data.db.user.driver")
- dsn := conf.GetString("data.db.user.dsn")
- // GORM doc: https://gorm.io/docs/connecting_to_the_database.html
- switch driver {
- case "mysql":
- db, err = gorm.Open(mysql.Open(dsn), &gorm.Config{
- Logger: logger,
- })
- case "postgres":
- db, err = gorm.Open(postgres.New(postgres.Config{
- DSN: dsn,
- PreferSimpleProtocol: true, // disables implicit prepared statement usage
- }), &gorm.Config{})
- case "sqlite":
- db, err = gorm.Open(sqlite.Open(dsn), &gorm.Config{})
- default:
- panic("unknown db driver")
- }
- if err != nil {
- panic(err)
- }
- db = db.Debug()
- // Connection Pool config
- sqlDB, err := db.DB()
- if err != nil {
- panic(err)
- }
- sqlDB.SetMaxIdleConns(10)
- sqlDB.SetMaxOpenConns(100)
- sqlDB.SetConnMaxLifetime(time.Hour)
- return db
- }
- func NewRedis(conf *viper.Viper) *redis.Client {
- rdb := redis.NewClient(&redis.Options{
- Addr: conf.GetString("data.redis.addr"),
- Password: conf.GetString("data.redis.password"),
- DB: conf.GetInt("data.redis.db"),
- })
- ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
- defer cancel()
- _, err := rdb.Ping(ctx).Result()
- if err != nil {
- panic(fmt.Sprintf("redis error: %s", err.Error()))
- }
- return rdb
- }
|