change.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. package tables
  2. import (
  3. "Wine-Server/utils"
  4. "database/sql"
  5. )
  6. type ChangeTable struct {
  7. Id uint32
  8. Wid uint32
  9. Device string
  10. Manager string
  11. Worker string
  12. Code string
  13. Status uint8
  14. Publish utils.TimeType
  15. Finish utils.TimeType
  16. Old [4]WineWithIdRemain
  17. New [4]WineWithIdRemain
  18. }
  19. func CreateChangeTable() error {
  20. SQL := "CREATE TABLE IF NOT EXISTS `change`(" +
  21. "`id` INT UNSIGNED AUTO_INCREMENT PRIMARY KEY NOT NULL," +
  22. "`wid` INT UNSIGNED NOT NULL," +
  23. "`device` VARCHAR(32) NOT NULL,`manager` VARCHAR(16) NOT NULL," +
  24. "`worker` VARCHAR(16) NOT NULL,`code` VARCHAR(6) NOT NULL," +
  25. "`status` SMALLINT UNSIGNED DEFAULT 0," +
  26. "`publish` DATETIME DEFAULT CURRENT_TIMESTAMP," +
  27. "`finish` DATETIME DEFAULT CURRENT_TIMESTAMP," +
  28. "`wine1o` SMALLINT UNSIGNED DEFAULT 0," +
  29. "`remain1o` SMALLINT UNSIGNED DEFAULT 0," +
  30. "`wine2o` SMALLINT UNSIGNED DEFAULT 0," +
  31. "`remain2o` SMALLINT UNSIGNED DEFAULT 0," +
  32. "`wine3o` SMALLINT UNSIGNED DEFAULT 0," +
  33. "`remain3o` SMALLINT UNSIGNED DEFAULT 0," +
  34. "`wine4o` SMALLINT UNSIGNED DEFAULT 0," +
  35. "`remain4o` SMALLINT UNSIGNED DEFAULT 0," +
  36. "`wine1n` SMALLINT UNSIGNED DEFAULT 0," +
  37. "`remain1n` SMALLINT UNSIGNED DEFAULT 0," +
  38. "`wine2n` SMALLINT UNSIGNED DEFAULT 0," +
  39. "`remain2n` SMALLINT UNSIGNED DEFAULT 0," +
  40. "`wine3n` SMALLINT UNSIGNED DEFAULT 0," +
  41. "`remain3n` SMALLINT UNSIGNED DEFAULT 0," +
  42. "`wine4n` SMALLINT UNSIGNED DEFAULT 0," +
  43. "`remain4n` SMALLINT UNSIGNED DEFAULT 0);"
  44. _, err := utils.Mysql.Exec(SQL)
  45. return err
  46. }
  47. func (row *ChangeTable) Insert() error {
  48. SQL := "INSERT INTO `change` (`wid`,`device`,`manager`,`worker`,`code`,`wine1o`,`remain1o`,`wine2o`,`remain2o`," +
  49. "`wine3o`,`remain3o`,`wine4o`,`remain4o`,`wine1n`,`remain1n`,`wine2n`,`remain2n`,`wine3n`,`remain3n`," +
  50. "`wine4n`,`remain4n`) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?);"
  51. pre, err := utils.Mysql.Prepare(SQL)
  52. if err != nil {
  53. return err
  54. }
  55. _, err = pre.Exec(
  56. row.Wid, row.Device, row.Manager, row.Worker, row.Code,
  57. row.Old[0].Id, row.Old[0].Remain, row.Old[1].Id, row.Old[1].Remain,
  58. row.Old[2].Id, row.Old[2].Remain, row.Old[3].Id, row.Old[3].Remain,
  59. row.New[0].Id, row.New[0].Remain, row.New[1].Id, row.New[1].Remain,
  60. row.New[2].Id, row.New[2].Remain, row.New[3].Id, row.New[3].Remain,
  61. )
  62. return err
  63. }
  64. func (row *ChangeTable) Delete() error {
  65. _, err := utils.Mysql.Exec("DELETE FROM `change` WHERE `id`=?;", row.Id)
  66. return err
  67. }
  68. func (row *ChangeTable) Update(args utils.JsonType) error {
  69. keys, values := utils.UnZip(args)
  70. SQL := utils.Format("UPDATE `change` SET %s WHERE `id`=%d;", utils.SqlFields(keys), row.Id)
  71. pre, err := utils.Mysql.Prepare(SQL)
  72. if err != nil {
  73. return err
  74. }
  75. _, err = pre.Exec(values...)
  76. return err
  77. }
  78. func (row *ChangeTable) UpdateSelf() error {
  79. SQL := "UPDATE `change` SET `wid`=?,`device`=?,`manager`=?,`worker`=?,`code`=?,`status`=?,`publish`=?,`finish`=?," +
  80. "`wine1o`=?,`remain1o`=?,`wine2o`=?,`remain2o`=?,`wine3o`=?,`remain3o`=?,`wine4o`=?,`remain4o`=?," +
  81. "`wine1n`=?,`remain1n`=?,`wine2n`=?,`remain2n`=?,`wine3n`=?,`remain3n`=?,`wine4n`=?,`remain4n`=? WHERE `id`=?;"
  82. _, err := utils.Mysql.Exec(
  83. SQL, row.Wid, row.Device, row.Manager, row.Worker,
  84. row.Code, row.Status, row.Publish, row.Finish,
  85. row.Old[0].Id, row.Old[0].Remain, row.Old[1].Id, row.Old[1].Remain,
  86. row.Old[2].Id, row.Old[2].Remain, row.Old[3].Id, row.Old[3].Remain,
  87. row.New[0].Id, row.New[0].Remain, row.New[1].Id, row.New[1].Remain,
  88. row.New[2].Id, row.New[2].Remain, row.New[3].Id, row.New[3].Remain, row.Id,
  89. )
  90. return err
  91. }
  92. func (row *ChangeTable) Get() error {
  93. SQL := "SELECT `wid`,`device`,`manager`,`worker`,`code`,`status`,`publish`,`finish`," +
  94. "`wine1o`,`remain1o`,`wine2o`,`remain2o`,`wine3o`,`remain3o`,`wine4o`,`remain4o`," +
  95. "`wine1n`,`remain1n`,`wine2n`,`remain2n`,`wine3n`,`remain3n`,`wine4n`,`remain4n` FROM `change` WHERE `id`=?;"
  96. err := utils.Mysql.QueryRow(SQL, row.Id).Scan(
  97. &row.Wid, &row.Device, &row.Manager, &row.Worker,
  98. &row.Code, &row.Status, &row.Publish, &row.Finish,
  99. &row.Old[0].Id, &row.Old[0].Remain, &row.Old[1].Id, &row.Old[1].Remain,
  100. &row.Old[2].Id, &row.Old[2].Remain, &row.Old[3].Id, &row.Old[3].Remain,
  101. &row.New[0].Id, &row.New[0].Remain, &row.New[1].Id, &row.New[1].Remain,
  102. &row.New[2].Id, &row.New[2].Remain, &row.New[3].Id, &row.New[3].Remain,
  103. )
  104. return err
  105. }
  106. type changeQueryRes struct {
  107. Id uint32 `json:"id"`
  108. Device struct {
  109. Id string `json:"id"`
  110. Addr string `json:"addr"`
  111. } `json:"device"`
  112. Code string `json:"code"`
  113. Status uint8 `json:"status"`
  114. Publish utils.TimeType `json:"publish"`
  115. Finish utils.TimeType `json:"finish"`
  116. Old [4]WineWithIdNameRemain `json:"old"`
  117. New [4]WineWithIdNameRemain `json:"new"`
  118. }
  119. func ChangesQuery(manager, worker, cond string, status, limit, page int) (int, []changeQueryRes, error) {
  120. SQL, like := "none", utils.Format("%%%s%%", cond)
  121. if status == -1 {
  122. SQL = utils.Format(
  123. "SELECT COUNT(c.`id`) FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id`"+
  124. "WHERE c.`manager`='%s' AND c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s')",
  125. manager, worker, like, like,
  126. )
  127. } else {
  128. SQL = utils.Format(
  129. "SELECT COUNT(c.`id`) FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id`"+
  130. "WHERE c.`status`=%d AND c.`manager`='%s' AND c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s')",
  131. status, manager, worker, like, like,
  132. )
  133. }
  134. total := 0
  135. err := utils.Mysql.QueryRow(SQL).Scan(&total)
  136. if err != nil {
  137. return 0, nil, err
  138. }
  139. if total == 0 {
  140. return 0, []changeQueryRes{}, err
  141. }
  142. if status == -1 {
  143. SQL = utils.Format(
  144. "SELECT c.`id`,d.`id`,d.`addr`,c.`code`,c.`status`,c.`publish`,c.`finish`,"+
  145. "w1.`id`,w1.`name`,c.`remain1o`,w2.`id`,w2.`name`,c.`remain2o`,"+
  146. "w3.`id`,w3.`name`,c.`remain3o`,w4.`id`,w4.`name`,c.`remain4o`,"+
  147. "w5.`id`,w5.`name`,c.`remain1n`,w6.`id`,w6.`name`,c.`remain2n`,"+
  148. "w7.`id`,w7.`name`,c.`remain3n`,w8.`id`,w8.`name`,c.`remain4n` "+
  149. "FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id`"+
  150. "INNER JOIN `wine` AS w1 ON w1.`id`=c.`wine1o` INNER JOIN `wine` AS w2 ON w2.`id`=c.`wine2o` "+
  151. "INNER JOIN `wine` AS w3 ON w3.`id`=c.`wine3o` INNER JOIN `wine` AS w4 ON w4.`id`=c.`wine4o` "+
  152. "INNER JOIN `wine` AS w5 ON w5.`id`=c.`wine1n` INNER JOIN `wine` AS w6 ON w6.`id`=c.`wine2n` "+
  153. "INNER JOIN `wine` AS w7 ON w7.`id`=c.`wine3n` INNER JOIN `wine` AS w8 ON w8.`id`=c.`wine4n` "+
  154. "WHERE c.`manager`='%s' AND c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s') "+
  155. "LIMIT %d OFFSET %d;", manager, worker, like, like, limit, (page-1)*limit,
  156. )
  157. } else {
  158. SQL = utils.Format(
  159. "SELECT c.`id`,d.`id`,d.`addr`,c.`code`,c.`status`,c.`publish`,c.`finish`,"+
  160. "w1.`id`,w1.`name`,c.`remain1o`,w2.`id`,w2.`name`,c.`remain2o`,"+
  161. "w3.`id`,w3.`name`,c.`remain3o`,w4.`id`,w4.`name`,c.`remain4o`,"+
  162. "w5.`id`,w5.`name`,c.`remain1n`,w6.`id`,w6.`name`,c.`remain2n`,"+
  163. "w7.`id`,w7.`name`,c.`remain3n`,w8.`id`,w8.`name`,c.`remain4n` "+
  164. "FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id`"+
  165. "INNER JOIN `wine` AS w1 ON w1.`id`=c.`wine1o` INNER JOIN `wine` AS w2 ON w2.`id`=c.`wine2o` "+
  166. "INNER JOIN `wine` AS w3 ON w3.`id`=c.`wine3o` INNER JOIN `wine` AS w4 ON w4.`id`=c.`wine4o` "+
  167. "INNER JOIN `wine` AS w5 ON w5.`id`=c.`wine1n` INNER JOIN `wine` AS w6 ON w6.`id`=c.`wine2n` "+
  168. "INNER JOIN `wine` AS w7 ON w7.`id`=c.`wine3n` INNER JOIN `wine` AS w8 ON w8.`id`=c.`wine4n` "+
  169. "WHERE c.`status`=%d AND c.`manager`='%s' AND c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s') "+
  170. "LIMIT %d OFFSET %d;", status, manager, worker, like, like, limit, (page-1)*limit,
  171. )
  172. }
  173. var rows *sql.Rows
  174. res := make([]changeQueryRes, 0)
  175. rows, err = utils.Mysql.Query(SQL)
  176. if err != nil {
  177. return 0, nil, err
  178. }
  179. for rows.Next() {
  180. var one changeQueryRes
  181. _ = rows.Scan(
  182. &one.Id, &one.Device.Id, &one.Device.Addr,
  183. &one.Code, &one.Status, &one.Publish, &one.Finish,
  184. &one.Old[0].Id, &one.Old[0].Name, &one.Old[0].Remain,
  185. &one.Old[1].Id, &one.Old[1].Name, &one.Old[1].Remain,
  186. &one.Old[2].Id, &one.Old[2].Name, &one.Old[2].Remain,
  187. &one.Old[3].Id, &one.Old[3].Name, &one.Old[3].Remain,
  188. &one.New[0].Id, &one.New[0].Name, &one.New[0].Remain,
  189. &one.New[1].Id, &one.New[1].Name, &one.New[1].Remain,
  190. &one.New[2].Id, &one.New[2].Name, &one.New[2].Remain,
  191. &one.New[3].Id, &one.New[3].Name, &one.New[3].Remain,
  192. )
  193. res = append(res, one)
  194. }
  195. _ = rows.Close()
  196. return total, res, nil
  197. }
  198. type changeQueryForWorkerRes struct {
  199. Id uint32 `json:"id"`
  200. Device struct {
  201. Id string `json:"id"`
  202. Addr string `json:"addr"`
  203. } `json:"device"`
  204. Status uint8 `json:"status"`
  205. Publish utils.TimeType `json:"publish"`
  206. Finish utils.TimeType `json:"finish"`
  207. }
  208. func ChangesQueryForWorker(worker, cond string, status, limit, page int) (int, []changeQueryForWorkerRes, error) {
  209. SQL, like := "none", utils.Format("%%%s%%", cond)
  210. if status == -1 {
  211. SQL = utils.Format(
  212. "SELECT COUNT(c.`id`) FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id`"+
  213. "WHERE c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s')",
  214. worker, like, like,
  215. )
  216. } else {
  217. SQL = utils.Format(
  218. "SELECT COUNT(c.`id`) FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id`"+
  219. "WHERE c.`status`=%d AND c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s')",
  220. status, worker, like, like,
  221. )
  222. }
  223. total := 0
  224. err := utils.Mysql.QueryRow(SQL).Scan(&total)
  225. if err != nil {
  226. return 0, nil, err
  227. }
  228. if total == 0 {
  229. return 0, []changeQueryForWorkerRes{}, err
  230. }
  231. if status == -1 {
  232. SQL = utils.Format(
  233. "SELECT c.`id`,d.`id`,d.`addr`,c.`status`,c.`publish`,c.`finish` "+
  234. "FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id` "+
  235. "WHERE c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s') "+
  236. "LIMIT %d OFFSET %d;", worker, like, like, limit, (page-1)*limit,
  237. )
  238. } else {
  239. SQL = utils.Format(
  240. "SELECT c.`id`,d.`id`,d.`addr`,c.`status`,c.`publish`,c.`finish` "+
  241. "FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id` "+
  242. "WHERE c.`status`=%d AND c.`worker`='%s' AND (d.`id` LIKE '%s' OR d.`addr` LIKE '%s') "+
  243. "LIMIT %d OFFSET %d;", status, worker, like, like, limit, (page-1)*limit,
  244. )
  245. }
  246. var rows *sql.Rows
  247. res := make([]changeQueryForWorkerRes, 0)
  248. rows, err = utils.Mysql.Query(SQL)
  249. if err != nil {
  250. return 0, nil, err
  251. }
  252. for rows.Next() {
  253. var one changeQueryForWorkerRes
  254. _ = rows.Scan(
  255. &one.Id, &one.Device.Id, &one.Device.Addr,
  256. &one.Status, &one.Publish, &one.Finish,
  257. )
  258. res = append(res, one)
  259. }
  260. _ = rows.Close()
  261. return total, res, nil
  262. }
  263. func ChangeDetail(cid uint32) (*changeQueryRes, error) {
  264. SQL := "SELECT c.`id`,d.`id`,d.`addr`,c.`code`,c.`status`,c.`publish`,c.`finish`," +
  265. "w1.`id`,w1.`name`,c.`remain1o`,w2.`id`,w2.`name`,c.`remain2o`," +
  266. "w3.`id`,w3.`name`,c.`remain3o`,w4.`id`,w4.`name`,c.`remain4o`," +
  267. "w5.`id`,w5.`name`,c.`remain1n`,w6.`id`,w6.`name`,c.`remain2n`," +
  268. "w7.`id`,w7.`name`,c.`remain3n`,w8.`id`,w8.`name`,c.`remain4n` " +
  269. "FROM `change` AS c LEFT JOIN `device` AS d ON c.`device`=d.`id`" +
  270. "INNER JOIN `wine` AS w1 ON w1.`id`=c.`wine1o` INNER JOIN `wine` AS w2 ON w2.`id`=c.`wine2o` " +
  271. "INNER JOIN `wine` AS w3 ON w3.`id`=c.`wine3o` INNER JOIN `wine` AS w4 ON w4.`id`=c.`wine4o` " +
  272. "INNER JOIN `wine` AS w5 ON w5.`id`=c.`wine1n` INNER JOIN `wine` AS w6 ON w6.`id`=c.`wine2n` " +
  273. "INNER JOIN `wine` AS w7 ON w7.`id`=c.`wine3n` INNER JOIN `wine` AS w8 ON w8.`id`=c.`wine4n` " +
  274. "WHERE c.`id`=?;"
  275. var res changeQueryRes
  276. err := utils.Mysql.QueryRow(SQL, cid).Scan(
  277. &res.Id, &res.Device.Id, &res.Device.Addr,
  278. &res.Code, &res.Status, &res.Publish, &res.Finish,
  279. &res.Old[0].Id, &res.Old[0].Name, &res.Old[0].Remain,
  280. &res.Old[1].Id, &res.Old[1].Name, &res.Old[1].Remain,
  281. &res.Old[2].Id, &res.Old[2].Name, &res.Old[2].Remain,
  282. &res.Old[3].Id, &res.Old[3].Name, &res.Old[3].Remain,
  283. &res.New[0].Id, &res.New[0].Name, &res.New[0].Remain,
  284. &res.New[1].Id, &res.New[1].Name, &res.New[1].Remain,
  285. &res.New[2].Id, &res.New[2].Name, &res.New[2].Remain,
  286. &res.New[3].Id, &res.New[3].Name, &res.New[3].Remain,
  287. )
  288. return &res, err
  289. }
  290. func ChangeRankWorkers(start, end string) ([]idNameTotal, error) {
  291. SQL := "SELECT w.`id` AS `person`,w.`name`,COUNT(c.`id`) AS `count` FROM `change` AS c " +
  292. "LEFT JOIN `worker` AS w ON c.`worker`=w.`id` WHERE c.`status`=2 " +
  293. "AND (c.`finish` BETWEEN TIMESTAMP(?) AND TIMESTAMP(?)) GROUP BY `person` ORDER BY `count` DESC;"
  294. rows, err := utils.Mysql.Query(SQL, start, end)
  295. if err != nil {
  296. return nil, err
  297. }
  298. res := make([]idNameTotal, 0)
  299. for rows.Next() {
  300. var one idNameTotal
  301. _ = rows.Scan(&one.Id, &one.Name, &one.Total)
  302. res = append(res, one)
  303. }
  304. _ = rows.Close()
  305. return res, nil
  306. }
  307. func ChangeRankWorkersForAdmin(manager, start, end string) ([]idNameTotal, error) {
  308. SQL := "SELECT w.`id` AS `person`,w.`name`,COUNT(c.`id`) AS `count` FROM `change` AS c " +
  309. "LEFT JOIN `worker` AS w ON c.`worker`=w.`id` WHERE c.`status`=2 AND c.`manager`=? " +
  310. "AND (c.`finish` BETWEEN TIMESTAMP(?) AND TIMESTAMP(?)) GROUP BY `person` ORDER BY `count` DESC;"
  311. rows, err := utils.Mysql.Query(SQL, manager, start, end)
  312. if err != nil {
  313. return nil, err
  314. }
  315. res := make([]idNameTotal, 0)
  316. for rows.Next() {
  317. var one idNameTotal
  318. _ = rows.Scan(&one.Id, &one.Name, &one.Total)
  319. res = append(res, one)
  320. }
  321. _ = rows.Close()
  322. return res, nil
  323. }
  324. func ChangeQueryWorker(id string, kind uint8) ([]timeTotal, error) {
  325. SQL := "none"
  326. if kind == 0 {
  327. SQL = "SELECT DATE_FORMAT(`finish`,'%Y-%m') as `tt`,COUNT(`id`) FROM `change` WHERE `status`=2 AND `worker`=? GROUP BY `tt`;"
  328. } else {
  329. align, days := getFmtAndDays(kind)
  330. SQL = utils.Format(
  331. "SELECT DATE_FORMAT(`finish`,'%s') as `tt`,COUNT(`tt`) FROM `change` WHERE `status`=2 AND `worker`=? AND (`finish` BETWEEN "+
  332. "DATE_SUB(CURRENT_TIMESTAMP,INTERVAL %d DAY) AND CURRENT_TIMESTAMP) GROUP BY `tt`;", align, days,
  333. )
  334. }
  335. rows, err := utils.Mysql.Query(SQL, id)
  336. if err != nil {
  337. return nil, err
  338. }
  339. res := make([]timeTotal, 0)
  340. for rows.Next() {
  341. var one timeTotal
  342. _ = rows.Scan(&one.Time, &one.Total)
  343. res = append(res, one)
  344. }
  345. _ = rows.Close()
  346. return res, nil
  347. }
  348. func ChangeQueryWorkerForAdmin(manager, id string, kind uint8) ([]timeTotal, error) {
  349. SQL := "none"
  350. if kind == 0 {
  351. SQL = "SELECT DATE_FORMAT(`finish`,'%Y-%m') as `tt`,COUNT(`id`) FROM `change` " +
  352. "WHERE `status`=2 AND `manager`=? AND `worker`=? GROUP BY `tt`;"
  353. } else {
  354. align, days := getFmtAndDays(kind)
  355. SQL = utils.Format(
  356. "SELECT DATE_FORMAT(`finish`,'%s') as `tt`,COUNT(`tt`) FROM `change` WHERE `status`=2 AND `manager`=? "+
  357. "AND `worker`=? AND (`finish` BETWEEN DATE_SUB(CURRENT_TIMESTAMP,INTERVAL %d DAY) AND CURRENT_TIMESTAMP) "+
  358. "GROUP BY `tt`;", align, days,
  359. )
  360. }
  361. rows, err := utils.Mysql.Query(SQL, manager, id)
  362. if err != nil {
  363. return nil, err
  364. }
  365. res := make([]timeTotal, 0)
  366. for rows.Next() {
  367. var one timeTotal
  368. _ = rows.Scan(&one.Time, &one.Total)
  369. res = append(res, one)
  370. }
  371. _ = rows.Close()
  372. return res, nil
  373. }