wp-go/model/sqxquery.go

155 lines
3.3 KiB
Go
Raw Normal View History

2023-02-06 09:58:24 +00:00
package model
import (
"context"
"fmt"
"github.com/fthvgb1/wp-go/helper/slice"
2023-02-06 09:58:24 +00:00
"github.com/jmoiron/sqlx"
"strconv"
"strings"
)
type SqlxQuery struct {
sqlx *sqlx.DB
UniversalDb
2023-02-06 09:58:24 +00:00
}
func NewSqlxQuery(sqlx *sqlx.DB, u UniversalDb) *SqlxQuery {
s := &SqlxQuery{sqlx: sqlx, UniversalDb: u}
if u.selects == nil {
s.UniversalDb.selects = s.Selects
}
if u.gets == nil {
s.UniversalDb.gets = s.Gets
}
return s
}
func SetSelect(db *SqlxQuery, fn func(context.Context, any, string, ...any) error) {
db.selects = fn
}
func SetGet(db *SqlxQuery, fn func(context.Context, any, string, ...any) error) {
db.gets = fn
2023-02-06 09:58:24 +00:00
}
func (r *SqlxQuery) Selects(ctx context.Context, dest any, sql string, params ...any) error {
2023-02-21 11:49:13 +00:00
v := ctx.Value("handle=>")
if v != nil {
vv, ok := v.(string)
if ok && vv != "" {
switch vv {
case "string":
return ToMapSlice(r.sqlx, dest.(*[]map[string]string), sql, params...)
2023-02-21 11:49:13 +00:00
case "scanner":
fn := ctx.Value("fn")
return Scanner[any](r.sqlx, dest, sql, params...)(fn.(func(any)))
}
}
2023-02-06 09:58:24 +00:00
}
return r.sqlx.Select(dest, sql, params...)
}
func (r *SqlxQuery) Gets(ctx context.Context, dest any, sql string, params ...any) error {
2023-02-21 11:49:13 +00:00
v := ctx.Value("handle=>")
if v != nil {
vv, ok := v.(string)
if ok && vv != "" {
switch vv {
case "string":
return GetToMap(r.sqlx, dest.(*map[string]string), sql, params...)
}
}
2023-02-06 09:58:24 +00:00
}
return r.sqlx.Get(dest, sql, params...)
}
2023-02-21 11:49:13 +00:00
func Scanner[T any](db *sqlx.DB, v T, s string, params ...any) func(func(T)) error {
return func(fn func(T)) error {
rows, err := db.Queryx(s, params...)
if err != nil {
return err
}
for rows.Next() {
err = rows.StructScan(v)
if err != nil {
return err
}
fn(v)
}
return nil
}
}
func ToMapSlice[V any](db *sqlx.DB, dest *[]map[string]V, sql string, params ...any) (err error) {
rows, err := db.Query(sql, params...)
columns, err := rows.Columns()
if err != nil {
return err
}
defer rows.Close()
columnLen := len(columns)
c := make([]*V, columnLen)
for i, _ := range c {
var a V
c[i] = &a
}
2023-02-21 11:49:13 +00:00
args := slice.ToAnySlice(c)
var m []map[string]V
for rows.Next() {
2023-02-21 11:49:13 +00:00
err = rows.Scan(args...)
if err != nil {
return
}
v := make(map[string]V)
for i, data := range c {
v[columns[i]] = *data
}
m = append(m, v)
}
*dest = m
return
}
func GetToMap[V any](db *sqlx.DB, dest *map[string]V, sql string, params ...any) (err error) {
rows := db.QueryRowx(sql, params...)
columns, err := rows.Columns()
if err != nil {
return err
}
columnLen := len(columns)
c := make([]*V, columnLen)
for i, _ := range c {
var a V
c[i] = &a
}
err = rows.Scan(slice.ToAnySlice(c)...)
if err != nil {
return
}
v := make(map[string]V)
for i, data := range c {
v[columns[i]] = *data
}
*dest = v
return
}
2023-02-10 13:23:30 +00:00
func FormatSql(sql string, params ...any) string {
2023-02-06 09:58:24 +00:00
for _, param := range params {
switch param.(type) {
case string:
sql = strings.Replace(sql, "?", fmt.Sprintf("'%s'", param.(string)), 1)
case int64:
sql = strings.Replace(sql, "?", strconv.FormatInt(param.(int64), 10), 1)
case int:
sql = strings.Replace(sql, "?", strconv.Itoa(param.(int)), 1)
case uint64:
sql = strings.Replace(sql, "?", strconv.FormatUint(param.(uint64), 10), 1)
case float64:
sql = strings.Replace(sql, "?", fmt.Sprintf("%f", param.(float64)), 1)
}
}
return sql
}