db.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. package db
  2. import (
  3. "database/sql"
  4. "errors"
  5. "fmt"
  6. "strconv"
  7. "time"
  8. "zhiyuan/pkg/config"
  9. "zhiyuan/pkg/logger"
  10. "zhiyuan/pkg/utils"
  11. "github.com/didi/gendry/builder"
  12. "github.com/didi/gendry/scanner"
  13. _ "github.com/go-sql-driver/mysql"
  14. )
  15. var db *sql.DB
  16. var GeoDB *sql.DB
  17. func Setup() {
  18. var err error
  19. db, err = sql.Open("mysql", fmt.Sprintf("%s:%s@tcp(%s)/%s?charset=%s&parseTime=True&loc=Local",
  20. config.Cfg.Database.User,
  21. config.Cfg.Database.Password,
  22. config.Cfg.Database.Host,
  23. config.Cfg.Database.Name,
  24. config.Cfg.Database.Charset))
  25. if err != nil {
  26. logger.Sugar.Error(err)
  27. }
  28. GeoDB, err = sql.Open("mysql", fmt.Sprintf("%s:%s@tcp(%s)/%s?charset=%s&parseTime=True&loc=Local",
  29. config.Cfg.Database.User,
  30. config.Cfg.Database.Password,
  31. config.Cfg.Database.Host,
  32. "geo",
  33. config.Cfg.Database.Charset))
  34. if err != nil {
  35. logger.Sugar.Error(err)
  36. }
  37. scanner.SetTagName("json")
  38. return
  39. }
  40. func Query(query string, params map[string]interface{}, retVal interface{}, dbs *sql.DB) error {
  41. if dbs == nil {
  42. dbs = db
  43. }
  44. cond, values, err := builder.NamedQuery(query, params)
  45. if err != nil {
  46. return err
  47. }
  48. logger.Sugar.Infof("query: %v; params: %v", cond, values)
  49. rows, err := dbs.Query(cond, values...)
  50. // defer rows.Close()
  51. if err != nil {
  52. logger.Sugar.Error(err)
  53. return err
  54. }
  55. return scanner.ScanClose(rows, retVal)
  56. }
  57. func resolveDataFromRows(rows *sql.Rows) ([]map[string]interface{}, error) {
  58. if nil == rows {
  59. return nil, errors.New("[scanner]: rows can't be nil")
  60. }
  61. columns, err := rows.Columns()
  62. if nil != err {
  63. return nil, err
  64. }
  65. length := len(columns)
  66. var result []map[string]interface{}
  67. //unnecessary to put below into rows.Next loop,reduce allocating
  68. values := make([]interface{}, length)
  69. for i := 0; i < length; i++ {
  70. values[i] = new(interface{})
  71. }
  72. for rows.Next() {
  73. err = rows.Scan(values...)
  74. if nil != err {
  75. return nil, err
  76. }
  77. mp := make(map[string]interface{})
  78. for idx, name := range columns {
  79. //mp[name] = reflect.ValueOf(values[idx]).Elem().Interface()
  80. mp[name] = *(values[idx].(*interface{}))
  81. }
  82. result = append(result, mp)
  83. }
  84. return result, nil
  85. }
  86. type CloseErr struct {
  87. err error
  88. }
  89. func (e CloseErr) Error() string {
  90. return e.err.Error()
  91. }
  92. func newCloseErr(err error) error {
  93. if err == nil {
  94. return nil
  95. }
  96. return CloseErr{err}
  97. }
  98. func QueryMap(query string, params map[string]interface{}, dbs *sql.DB) ([]map[string]interface{}, error) {
  99. if dbs == nil {
  100. dbs = db
  101. }
  102. cond, values, err := builder.NamedQuery(query, params)
  103. if err != nil {
  104. return nil, err
  105. }
  106. //logger.Sugar.Infof("query: %v; params: %v", cond, values)
  107. rows, err := dbs.Query(cond, values...)
  108. // defer rows.Close()
  109. if err != nil {
  110. logger.Sugar.Error(err)
  111. return nil, err
  112. }
  113. //result, err := scanner.ScanMapDecodeClose(rows)
  114. results, err := resolveDataFromRows(rows)
  115. if nil != err {
  116. return nil, err
  117. }
  118. for i := 0; i < len(results); i++ {
  119. for k, v := range results[i] {
  120. rv, ok := v.([]uint8)
  121. if !ok {
  122. continue
  123. }
  124. s := string(rv)
  125. results[i][k] = s
  126. if len(s) > 8 {
  127. continue
  128. }
  129. //convert to int
  130. intVal, err := strconv.Atoi(s)
  131. if err == nil {
  132. results[i][k] = intVal
  133. continue
  134. }
  135. //convert to float64
  136. floatVal, err := strconv.ParseFloat(s, 64)
  137. if err == nil {
  138. results[i][k] = floatVal
  139. continue
  140. }
  141. }
  142. }
  143. if nil != rows {
  144. errClose := rows.Close()
  145. if err == nil {
  146. err = newCloseErr(errClose)
  147. }
  148. }
  149. return results, err
  150. }
  151. func BuildInsert(table string, data []map[string]interface{}, dbs *sql.DB) (int64, error) {
  152. if dbs == nil {
  153. dbs = db
  154. }
  155. cond, values, err := builder.BuildInsert(table, data)
  156. if nil != err {
  157. logger.Sugar.Error(err)
  158. return 0, err
  159. }
  160. //logger.Sugar.Infof("query: %v; params: %v", cond, values)
  161. result, err := dbs.Exec(cond, values...)
  162. if nil != err || nil == result {
  163. logger.Sugar.Error(err)
  164. return 0, err
  165. }
  166. return result.LastInsertId()
  167. }
  168. func BuildUpdate(table string, where, data map[string]interface{}, dbs *sql.DB) (int64, error) {
  169. if dbs == nil {
  170. dbs = db
  171. }
  172. cond, values, err := builder.BuildUpdate(table, where, data)
  173. if nil != err {
  174. return 0, err
  175. }
  176. //logger.Sugar.Infof("query: %v; params: %v", cond, values)
  177. result, err := dbs.Exec(cond, values...)
  178. if nil != err || nil == result {
  179. logger.Sugar.Error(err)
  180. return 0, err
  181. }
  182. return result.RowsAffected()
  183. }
  184. func GetMultiMap(table string, where map[string]interface{}, fields []string) ([]map[string]interface{}, error) {
  185. rows, err := BuildSelectRows(table, where, fields)
  186. if err != nil {
  187. return nil, err
  188. }
  189. result, mapErr := scanner.ScanMapDecodeClose(rows)
  190. return result, mapErr
  191. }
  192. func GetMulti(table string, where map[string]interface{}, fields []string, retVal interface{}) error {
  193. rows, err := BuildSelectRows(table, where, fields)
  194. if err != nil {
  195. return err
  196. }
  197. return ScanClose(rows, retVal)
  198. }
  199. func GetOneMap(table string, where map[string]interface{}, fields []string) (map[string]interface{}, error) {
  200. wheres := make(map[string]interface{})
  201. for key, value := range where {
  202. wheres[key] = value
  203. }
  204. wheres["_limit"] = []uint{1}
  205. result, err := GetMultiMap(table, wheres, fields)
  206. if err != nil {
  207. return nil, err
  208. }
  209. if len(result) == 0 {
  210. return nil, nil
  211. }
  212. return result[0], nil
  213. }
  214. func GetOne(table string, where map[string]interface{}, fields []string, retVal interface{}) error {
  215. wheres := make(map[string]interface{})
  216. for key, value := range where {
  217. wheres[key] = value
  218. }
  219. wheres["_limit"] = []uint{1}
  220. return GetMulti(table, where, fields, retVal)
  221. }
  222. func ScanClose(rows *sql.Rows, target interface{}) error {
  223. err := scanner.ScanClose(rows, target)
  224. if err == scanner.ErrEmptyResult {
  225. return nil
  226. }
  227. return err
  228. }
  229. func Count(table string, where map[string]interface{}) (int64, error) {
  230. res, err := GetOneMap(table, where, []string{"count(*) as count"})
  231. if err != nil {
  232. return 0, err
  233. }
  234. return res["count"].(int64), err
  235. }
  236. func BuildSelectRows(table string, where map[string]interface{}, fields []string) (*sql.Rows, error) {
  237. cond, values, err := BuildSelect(table, where, fields)
  238. if err != nil {
  239. return nil, err
  240. }
  241. return GetRows(cond, values)
  242. }
  243. func BuildSelect(table string, where map[string]interface{}, fields []string) (string, []interface{}, error) {
  244. cond, values, err := builder.BuildSelect(table, where, fields)
  245. if err != nil {
  246. logger.Sugar.Error(err)
  247. }
  248. return cond, values, err
  249. }
  250. func GetRows(cond string, values []interface{}) (*sql.Rows, error) {
  251. logger.Sugar.Infof("query: %v; params: %v", cond, values)
  252. rows, err := db.Query(cond, values...)
  253. // defer rows.Close()
  254. if err != nil {
  255. logger.Sugar.Error(err)
  256. return nil, errors.New("未知错误")
  257. }
  258. return rows, nil
  259. }
  260. func CountRaw(query string, params map[string]interface{}) (int64, error) {
  261. res, err := GetOneMapRaw("SELECT count(*) as count FROM "+query, params)
  262. if err != nil {
  263. return 0, err
  264. }
  265. return utils.ToInt64(res["count"]), err
  266. }
  267. func GetMultiRaw(query string, where map[string]string, params map[string]interface{}, retVal interface{}) error {
  268. cond, values, err := builder.NamedQuery(JoinQuery(query, where), params)
  269. if err != nil {
  270. return err
  271. }
  272. rows, err := GetRows(cond, values)
  273. if err != nil {
  274. return err
  275. }
  276. return scanner.ScanClose(rows, retVal)
  277. }
  278. func JoinQuery(query string, where map[string]string) string {
  279. if where["where"] != "" {
  280. query = query + " WHERE " + where["where"]
  281. }
  282. if where["_group_by"] != "" {
  283. query = query + " GROUP BY " + where["_group_by"]
  284. }
  285. if where["_having"] != "" {
  286. query = query + " HAVING " + where["_having"]
  287. }
  288. if where["_order_by"] != "" {
  289. query = query + " ORDER BY " + where["_order_by"]
  290. }
  291. if where["_page_size"] != "" && where["_page_num"] != "" {
  292. pageNum := utils.ToInt(where["_page_num"])
  293. pageSize := utils.ToInt(where["_page_size"])
  294. query = query + " LIMIT " + utils.ToStr((pageNum-1)*pageSize) + "," + utils.ToStr(pageSize)
  295. }
  296. return query
  297. }
  298. func GetMultiMapRaw(query string, params map[string]interface{}) ([]map[string]interface{}, error) {
  299. cond, values, err := builder.NamedQuery(query, params)
  300. if err != nil {
  301. return nil, err
  302. }
  303. rows, err := GetRows(cond, values)
  304. if err != nil {
  305. return nil, err
  306. }
  307. result, mapErr := scanner.ScanMapDecodeClose(rows)
  308. return result, mapErr
  309. }
  310. func GetOneMapRaw(query string, params map[string]interface{}) (map[string]interface{}, error) {
  311. result, err := GetMultiMapRaw(query, params)
  312. if err != nil {
  313. return nil, err
  314. }
  315. if len(result) == 0 {
  316. return nil, errors.New("empty record1")
  317. }
  318. return result[0], err
  319. }
  320. func UpdateRaw(query string, params map[string]interface{}) (int64, error) {
  321. cond, values, err := builder.NamedQuery(query, params)
  322. if err != nil {
  323. return 0, err
  324. }
  325. return GetUpdateResult(cond, values)
  326. }
  327. func UpdateRawSimple(table string, querySet, queryWhere string, params map[string]interface{}) (int64, error) {
  328. querySet = querySet + ",updated_at=" + utils.ToStr(time.Now().Unix())
  329. query := fmt.Sprintf("UPDATE %s SET %s WHERE %s", table, querySet, queryWhere)
  330. return UpdateRaw(query, params)
  331. }
  332. func InsertRaw(query string, params map[string]interface{}) (int64, error) {
  333. cond, values, err := builder.NamedQuery(query, params)
  334. if err != nil {
  335. return 0, err
  336. }
  337. return GetInsertResult(cond, values)
  338. }
  339. func Update(table string, where, data map[string]interface{}) (int64, error) {
  340. data["updated_at"] = time.Now().Unix()
  341. cond, values, err := builder.BuildUpdate(table, where, data)
  342. if nil != err {
  343. return 0, err
  344. }
  345. return GetUpdateResult(cond, values)
  346. }
  347. func Delete(table string, where map[string]interface{}) (int64, error) {
  348. cond, values, err := builder.BuildDelete(table, where)
  349. if nil != err {
  350. return 0, err
  351. }
  352. return GetUpdateResult(cond, values)
  353. }
  354. func DeleteSoft(table string, where map[string]interface{}) (int64, error) {
  355. cond, values, err := builder.BuildUpdate(table, where, map[string]interface{}{"deleted_at": time.Now().Unix()})
  356. if nil != err {
  357. return 0, err
  358. }
  359. return GetUpdateResult(cond, values)
  360. }
  361. func Insert(table string, data []map[string]interface{}) (int64, error) {
  362. for k, v := range data {
  363. v["created_at"] = time.Now().Unix()
  364. v["updated_at"] = time.Now().Unix()
  365. data[k] = v
  366. }
  367. cond, values, err := builder.BuildInsert(table, data)
  368. if nil != err {
  369. logger.Sugar.Error(err)
  370. return 0, err
  371. }
  372. return GetInsertResult(cond, values)
  373. }
  374. func InsertOne(table string, data map[string]interface{}) (int64, error) {
  375. return Insert(table, []map[string]interface{}{data})
  376. }
  377. func GetUpdateResult(cond string, values []interface{}) (int64, error) {
  378. result, err := Exec(cond, values)
  379. if err != nil {
  380. return 0, err
  381. }
  382. return result.RowsAffected()
  383. }
  384. func GetInsertResult(cond string, values []interface{}) (int64, error) {
  385. result, err := Exec(cond, values)
  386. if err != nil {
  387. return 0, err
  388. }
  389. return result.LastInsertId()
  390. }
  391. func Exec(cond string, values []interface{}) (sql.Result, error) {
  392. //logger.Sugar.Infof("query: %v; params: %v", cond, values)
  393. result, err := db.Exec(cond, values...)
  394. if nil != err || nil == result {
  395. logger.Sugar.Error(err)
  396. return nil, errors.New("未知错误")
  397. }
  398. return result, err
  399. }
  400. func GetOffset(pageNum uint, pageSize uint) []uint {
  401. return []uint{(pageNum - 1) * pageSize, pageSize}
  402. }