mm_os 3.3.1 → 4.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (380) hide show
  1. package/LICENSE +21 -201
  2. package/README.md +498 -99
  3. package/README_EN.md +505 -0
  4. package/adapter/adapter.js +431 -0
  5. package/adapter/custom_persistence.js +660 -0
  6. package/adapter/mqtt.js +273 -0
  7. package/adapter/socket.js +113 -0
  8. package/adapter/web.js +67 -0
  9. package/adapter/websocket.js +146 -0
  10. package/com/api/com.json +5 -0
  11. package/{core/com → com}/api/config.tpl.json +8 -8
  12. package/com/api/drive.js +708 -0
  13. package/com/api/index.js +198 -0
  14. package/com/api/oauth.js +200 -0
  15. package/com/api/script.tpl.js +32 -0
  16. package/com/cmd/README.md +11 -0
  17. package/com/cmd/com.json +5 -0
  18. package/com/cmd/config.tpl.json +122 -0
  19. package/com/cmd/drive.js +1548 -0
  20. package/com/cmd/index.js +1066 -0
  21. package/com/cmd/msg.json +48 -0
  22. package/com/cmd/nlp.js +525 -0
  23. package/com/cmd/script.tpl.js +32 -0
  24. package/com/db/com.json +5 -0
  25. package/com/db/drive.js +1999 -0
  26. package/com/db/index.js +242 -0
  27. package/{core/com → com}/event/README.md +4 -4
  28. package/com/event/com.json +5 -0
  29. package/{core/com → com}/event/config.tpl.json +18 -18
  30. package/com/event/drive.js +59 -0
  31. package/com/event/index.js +409 -0
  32. package/com/event/script.tpl.js +23 -0
  33. package/com/mqtt/com.json +5 -0
  34. package/{core/com → com}/mqtt/config.tpl.json +3 -5
  35. package/com/mqtt/drive.js +676 -0
  36. package/com/mqtt/index.js +822 -0
  37. package/com/mqtt/mm_mqtt.js +425 -0
  38. package/com/mqtt/script.tpl.js +723 -0
  39. package/com/nav/com.json +5 -0
  40. package/com/nav/config.tpl.json +84 -0
  41. package/com/nav/drive.js +702 -0
  42. package/com/nav/index.js +231 -0
  43. package/{core/com → com}/nav/tpl/admin_pc/page_config.vue +280 -280
  44. package/{core/com → com}/nav/tpl/admin_pc/page_config_form.vue +194 -194
  45. package/com/nav/tpl/admin_pc/page_form.vue +180 -0
  46. package/com/nav/tpl/admin_pc/page_view.vue +124 -0
  47. package/com/nav/tpl/dev_pc/page_default.vue +247 -0
  48. package/com/nav/tpl/dev_pc/page_type.vue +313 -0
  49. package/com/nav/tpl/home_pc/page_default.vue +234 -0
  50. package/com/nav/tpl/home_pc/page_form.vue +137 -0
  51. package/com/nav/tpl/home_pc/page_list.vue +234 -0
  52. package/com/nav/tpl/home_pc/page_nav.vue +221 -0
  53. package/com/nav/tpl/home_pc/page_type.vue +234 -0
  54. package/com/nav/tpl/home_pc/page_view.vue +125 -0
  55. package/com/nav/tpl/home_phone/page_channel.vue +234 -0
  56. package/com/nav/tpl/home_phone/page_default.vue +234 -0
  57. package/com/nav/tpl/home_phone/page_form.vue +137 -0
  58. package/com/nav/tpl/home_phone/page_nav.vue +237 -0
  59. package/com/nav/tpl/home_phone/page_type.vue +234 -0
  60. package/com/nav/tpl/home_phone/page_view.vue +125 -0
  61. package/com/nav/viewmodel.js +446 -0
  62. package/com/param/com.json +5 -0
  63. package/{core/com → com}/param/config.tpl.json +7 -1
  64. package/com/param/drive.js +502 -0
  65. package/com/param/index.js +155 -0
  66. package/com/param/script.tpl.js +12 -0
  67. package/com/pendant/com.json +5 -0
  68. package/{core/com/component → com/pendant}/config.tpl.json +15 -13
  69. package/com/pendant/drive.js +204 -0
  70. package/com/pendant/index.js +441 -0
  71. package/com/pendant/pendant.html +16 -0
  72. package/com/pendant/script.tpl.js +18 -0
  73. package/com/socket/com.json +5 -0
  74. package/com/socket/config.tpl.json +12 -0
  75. package/com/socket/drive.js +651 -0
  76. package/com/socket/index.js +351 -0
  77. package/com/socket/script.tpl.js +41 -0
  78. package/com/sql/com.json +5 -0
  79. package/{core/com → com}/sql/config.tpl.json +13 -9
  80. package/com/sql/drive.js +1259 -0
  81. package/com/sql/index.js +150 -0
  82. package/com/sql/script.tpl.js +47 -0
  83. package/com/static/com.json +5 -0
  84. package/{core/com → com}/static/config.tpl.json +10 -6
  85. package/com/static/drive.js +194 -0
  86. package/com/static/index.js +226 -0
  87. package/com/static/script.tpl.js +28 -0
  88. package/com/task/com.json +5 -0
  89. package/{core/com → com}/task/config.tpl.json +4 -6
  90. package/com/task/drive.js +405 -0
  91. package/com/task/index.js +148 -0
  92. package/com/task/script.tpl.js +37 -0
  93. package/com/template/com.json +5 -0
  94. package/com/template/config.tpl.json +16 -0
  95. package/com/template/drive.js +80 -0
  96. package/com/template/index.js +141 -0
  97. package/com.js +156 -0
  98. package/common/README.md +2 -0
  99. package/common/handler/msg/handler.json +22 -0
  100. package/common/handler/msg/index.js +23 -0
  101. package/common/handler/player/handler.json +22 -0
  102. package/common/handler/player/index.js +287 -0
  103. package/common/handler/user/handler.json +22 -0
  104. package/common/handler/user/index.js +23 -0
  105. package/common/middleware/web_after/index.js +29 -0
  106. package/common/middleware/web_after/middleware.json +9 -0
  107. package/common/middleware/web_base/index.js +113 -0
  108. package/common/middleware/web_base/middleware.json +19 -0
  109. package/common/middleware/web_before/index.js +33 -0
  110. package/common/middleware/web_before/middleware.json +9 -0
  111. package/common/middleware/web_cors/index.js +87 -0
  112. package/common/middleware/web_cors/middleware.json +24 -0
  113. package/common/middleware/web_error/index.js +119 -0
  114. package/common/middleware/web_error/middleware.json +18 -0
  115. package/common/middleware/web_ip/index.js +15 -0
  116. package/common/middleware/web_ip/middleware.json +14 -0
  117. package/common/middleware/web_logger/index.js +156 -0
  118. package/common/middleware/web_logger/middleware.json +14 -0
  119. package/common/middleware/web_main/index.js +24 -0
  120. package/common/middleware/web_main/middleware.json +9 -0
  121. package/common/middleware/web_static/index.js +73 -0
  122. package/common/middleware/web_static/middleware.json +54 -0
  123. package/common/middleware/web_waf/index.js +385 -0
  124. package/common/middleware/web_waf/middleware.json +13 -0
  125. package/common/model/msg/index.js +88 -0
  126. package/common/model/msg/model.json +401 -0
  127. package/common/model/player/index.js +63 -0
  128. package/common/model/player/model.json +185 -0
  129. package/common/model/user/index.js +11 -0
  130. package/common/model/user/model.json +219 -0
  131. package/core/app/config.tpl.json +67 -0
  132. package/core/app/index.js +632 -0
  133. package/core/app/script.tpl.js +52 -0
  134. package/core/channel/index.js +899 -0
  135. package/core/channel/matcher.js +585 -0
  136. package/core/com/config.tpl.json +16 -0
  137. package/core/com/index.js +74 -0
  138. package/core/com/script.tpl.js +5 -0
  139. package/core/component/component.js +42 -0
  140. package/core/component/config.tpl.json +63 -0
  141. package/core/component/index.js +273 -0
  142. package/core/component/script.tpl.js +19 -0
  143. package/core/controller/config.tpl.json +14 -0
  144. package/core/controller/index.js +373 -0
  145. package/core/controller/script.tpl.js +27 -0
  146. package/core/factory/config.tpl.json +14 -0
  147. package/core/factory/entity.js +275 -0
  148. package/core/factory/index.js +241 -0
  149. package/core/factory/script.tpl.js +16 -0
  150. package/core/game/bat/index.js +137 -0
  151. package/core/game/bat/world.js +622 -0
  152. package/core/game/config.tpl.json +16 -0
  153. package/core/game/entity_admin.js +230 -0
  154. package/core/game/index.js +186 -0
  155. package/core/handler/config.tpl.json +22 -0
  156. package/core/handler/index.js +181 -0
  157. package/core/handler/script.tpl.js +23 -0
  158. package/core/logic/config.tpl.json +14 -0
  159. package/core/logic/index.js +59 -0
  160. package/core/logic/script.tpl.js +19 -0
  161. package/core/middleware/config.tpl.json +16 -0
  162. package/core/middleware/index.js +125 -0
  163. package/core/middleware/script.tpl.js +37 -0
  164. package/core/mod/config.tpl.json +22 -0
  165. package/core/mod/index.js +130 -0
  166. package/core/mod/script.tpl.js +34 -0
  167. package/core/model/config.tpl.json +219 -0
  168. package/core/model/index.js +272 -0
  169. package/core/model/model.js +27 -0
  170. package/core/model/script.tpl.js +20 -0
  171. package/core/notifier/config.tpl.json +14 -0
  172. package/core/notifier/index.js +77 -0
  173. package/core/notifier/script.tpl.js +20 -0
  174. package/core/plugin/config.tpl.json +24 -0
  175. package/core/plugin/index.js +232 -0
  176. package/core/plugin/script.tpl.js +51 -0
  177. package/core/pusher/config.tpl.json +14 -0
  178. package/core/pusher/index.js +161 -0
  179. package/core/pusher/script.tpl.js +20 -0
  180. package/core/room/bat/index.js +170 -0
  181. package/core/room/bat/room.js +524 -0
  182. package/core/room/config.tpl.json +20 -0
  183. package/core/room/index.js +249 -0
  184. package/core/room/room.js +61 -0
  185. package/core/scene/config.tpl.json +14 -0
  186. package/core/scene/index.js +466 -0
  187. package/core/scene/loop.js +1255 -0
  188. package/core/scene/map.js +28 -0
  189. package/core/scene/script.tpl.js +22 -0
  190. package/core/sender/config.tpl.json +14 -0
  191. package/core/sender/index.js +79 -0
  192. package/core/sender/script.tpl.js +20 -0
  193. package/core/service/config.tpl.json +14 -0
  194. package/core/service/index.js +100 -0
  195. package/core/service/script.tpl.js +25 -0
  196. package/core/store/config.tpl.json +26 -0
  197. package/core/store/index.js +1755 -0
  198. package/core/store/script.tpl.js +22 -0
  199. package/core/store/sql.js +1464 -0
  200. package/core/system/config.tpl.json +18 -0
  201. package/core/system/index.js +312 -0
  202. package/core/system/script.tpl.js +77 -0
  203. package/core/view/config.tpl.json +14 -0
  204. package/core/view/index.js +91 -0
  205. package/core/view/script.tpl.js +20 -0
  206. package/core/zone/bat/index.js +725 -0
  207. package/core/zone/config.tpl.json +54 -0
  208. package/core/zone/index.js +614 -0
  209. package/core/zone/script.tpl.js +10 -0
  210. package/core/zone/zone_bat.js +136 -0
  211. package/core//345/237/272/347/261/273/346/250/241/345/235/227/346/270/205/345/215/225.md +24 -0
  212. package/index.js +17 -333
  213. package/os.js +57 -0
  214. package/package.json +65 -55
  215. package/server.js +598 -0
  216. package/README.en.md +0 -36
  217. package/conf.json +0 -3
  218. package/core/base/mqtt/index.js +0 -1110
  219. package/core/base/mqtt/lib.js +0 -40
  220. package/core/base/web/index.js +0 -245
  221. package/core/com/api/com.json +0 -4
  222. package/core/com/api/drive.js +0 -668
  223. package/core/com/api/index.js +0 -108
  224. package/core/com/api/oauth.js +0 -158
  225. package/core/com/api/script.js +0 -32
  226. package/core/com/app/README.md +0 -3
  227. package/core/com/app/com.json +0 -4
  228. package/core/com/app/config.tpl.json +0 -16
  229. package/core/com/app/drive.js +0 -309
  230. package/core/com/app/index.js +0 -211
  231. package/core/com/app/script.js +0 -155
  232. package/core/com/cmd/com.json +0 -4
  233. package/core/com/cmd/config.tpl.json +0 -66
  234. package/core/com/cmd/drive.js +0 -513
  235. package/core/com/cmd/index.js +0 -354
  236. package/core/com/cmd/old/5w2h.js +0 -54
  237. package/core/com/cmd/old/drive.js +0 -423
  238. package/core/com/cmd/script.js +0 -11
  239. package/core/com/component/README.md +0 -3
  240. package/core/com/component/com.json +0 -4
  241. package/core/com/component/component.html +0 -16
  242. package/core/com/component/drive.js +0 -197
  243. package/core/com/component/index.js +0 -312
  244. package/core/com/component/script.js +0 -18
  245. package/core/com/db/com.json +0 -4
  246. package/core/com/db/drive.js +0 -1160
  247. package/core/com/db/index.js +0 -176
  248. package/core/com/event/com.json +0 -4
  249. package/core/com/event/drive.js +0 -133
  250. package/core/com/event/index.js +0 -345
  251. package/core/com/event/script.js +0 -26
  252. package/core/com/eventer/com.js +0 -477
  253. package/core/com/eventer/com.json +0 -4
  254. package/core/com/middleware/com.js +0 -154
  255. package/core/com/middleware/com.json +0 -4
  256. package/core/com/middleware/config.tpl.json +0 -8
  257. package/core/com/middleware/script.js +0 -9
  258. package/core/com/mqtt/com.json +0 -4
  259. package/core/com/mqtt/drive.js +0 -600
  260. package/core/com/mqtt/index.js +0 -572
  261. package/core/com/mqtt/mm_mqtt.js +0 -330
  262. package/core/com/mqtt/script.js +0 -604
  263. package/core/com/msg/com.js +0 -296
  264. package/core/com/msg/com.json +0 -4
  265. package/core/com/nav/com.json +0 -4
  266. package/core/com/nav/config.tpl.json +0 -75
  267. package/core/com/nav/drive.js +0 -549
  268. package/core/com/nav/index.js +0 -182
  269. package/core/com/nav/tpl/admin_pc/page_form.vue +0 -180
  270. package/core/com/nav/tpl/admin_pc/page_view.vue +0 -124
  271. package/core/com/nav/tpl/dev_pc/page_default.vue +0 -247
  272. package/core/com/nav/tpl/dev_pc/page_type.vue +0 -313
  273. package/core/com/nav/tpl/home_pc/page_default.vue +0 -234
  274. package/core/com/nav/tpl/home_pc/page_form.vue +0 -137
  275. package/core/com/nav/tpl/home_pc/page_list.vue +0 -234
  276. package/core/com/nav/tpl/home_pc/page_nav.vue +0 -221
  277. package/core/com/nav/tpl/home_pc/page_type.vue +0 -234
  278. package/core/com/nav/tpl/home_pc/page_view.vue +0 -125
  279. package/core/com/nav/tpl/home_phone/page_channel.vue +0 -234
  280. package/core/com/nav/tpl/home_phone/page_default.vue +0 -234
  281. package/core/com/nav/tpl/home_phone/page_form.vue +0 -137
  282. package/core/com/nav/tpl/home_phone/page_nav.vue +0 -237
  283. package/core/com/nav/tpl/home_phone/page_type.vue +0 -234
  284. package/core/com/nav/tpl/home_phone/page_view.vue +0 -125
  285. package/core/com/nav/viewmodel.js +0 -296
  286. package/core/com/param/drive.js +0 -366
  287. package/core/com/param/index.js +0 -80
  288. package/core/com/param/script.js +0 -12
  289. package/core/com/param/test.js +0 -98
  290. package/core/com/plugin/README.md +0 -3
  291. package/core/com/plugin/com.json +0 -4
  292. package/core/com/plugin/config.tpl.json +0 -26
  293. package/core/com/plugin/drive.js +0 -536
  294. package/core/com/plugin/index.js +0 -259
  295. package/core/com/plugin/script.js +0 -213
  296. package/core/com/rpc/com.json +0 -4
  297. package/core/com/rpc/drive.js +0 -160
  298. package/core/com/rpc/index.js +0 -87
  299. package/core/com/rpc/rpc.js +0 -118
  300. package/core/com/socket/com.json +0 -4
  301. package/core/com/socket/config.tpl.json +0 -14
  302. package/core/com/socket/drive.js +0 -403
  303. package/core/com/socket/index.js +0 -62
  304. package/core/com/socket/script.js +0 -42
  305. package/core/com/sql/drive.js +0 -1087
  306. package/core/com/sql/index.js +0 -83
  307. package/core/com/sql/script.js +0 -48
  308. package/core/com/static/com.json +0 -4
  309. package/core/com/static/drive.js +0 -220
  310. package/core/com/static/index.js +0 -149
  311. package/core/com/static/script.js +0 -28
  312. package/core/com/task/com.json +0 -4
  313. package/core/com/task/drive.js +0 -403
  314. package/core/com/task/index.js +0 -110
  315. package/core/com/task/script.js +0 -37
  316. package/core/com/timer/com.js +0 -217
  317. package/core/com/timer/com.json +0 -4
  318. package/core/com/tpl/com.js +0 -19
  319. package/core/com/tpl/com.json +0 -4
  320. package/lib/actions.js +0 -50
  321. package/lib/base.js +0 -361
  322. package/lib/com.js +0 -29
  323. package/lib/ref.js +0 -121
  324. package/middleware/cors/index.js +0 -119
  325. package/middleware/cors/middleware.json +0 -20
  326. package/middleware/csrf/index.js +0 -202
  327. package/middleware/csrf/middleware.json +0 -24
  328. package/middleware/ip_firewall/index.js +0 -476
  329. package/middleware/ip_firewall/middleware.json +0 -109
  330. package/middleware/mqtt_base/index.js +0 -10
  331. package/middleware/mqtt_base/middleware.json +0 -11
  332. package/middleware/security_audit/index.js +0 -543
  333. package/middleware/security_audit/middleware.json +0 -48
  334. package/middleware/waf/index.js +0 -343
  335. package/middleware/waf/middleware.json +0 -10
  336. package/middleware/waf_ddos/index.js +0 -520
  337. package/middleware/waf_ddos/middleware.json +0 -38
  338. package/middleware/waf_xss/index.js +0 -269
  339. package/middleware/waf_xss/middleware.json +0 -18
  340. package/middleware/web_after/index.js +0 -33
  341. package/middleware/web_after/middleware.json +0 -10
  342. package/middleware/web_base/index.js +0 -90
  343. package/middleware/web_base/middleware.json +0 -10
  344. package/middleware/web_before/index.js +0 -27
  345. package/middleware/web_before/middleware.json +0 -10
  346. package/middleware/web_check/index.js +0 -28
  347. package/middleware/web_check/middleware.json +0 -10
  348. package/middleware/web_main/index.js +0 -28
  349. package/middleware/web_main/middleware.json +0 -10
  350. package/middleware/web_proxy/index.js +0 -37
  351. package/middleware/web_proxy/middleware.json +0 -10
  352. package/middleware/web_render/index.js +0 -87
  353. package/middleware/web_render/middleware.json +0 -10
  354. package/middleware/web_socket/index.js +0 -34
  355. package/middleware/web_socket/middleware.json +0 -10
  356. package/middleware/web_static/index.js +0 -115
  357. package/middleware/web_static/middleware.json +0 -10
  358. /package/{core/com → com}/api/README.md +0 -0
  359. /package/{core/com → com}/db/README.md +0 -0
  360. /package/{core/com → com}/mqtt/README.md +0 -0
  361. /package/{core/com → com}/nav/README.md +0 -0
  362. /package/{core/com → com}/nav/tpl/admin_pc/page_channel.vue +0 -0
  363. /package/{core/com → com}/nav/tpl/admin_pc/page_default.vue +0 -0
  364. /package/{core/com → com}/nav/tpl/admin_pc/page_lang.vue +0 -0
  365. /package/{core/com → com}/nav/tpl/admin_pc/page_nav.vue +0 -0
  366. /package/{core/com → com}/nav/tpl/admin_pc/page_table.vue +0 -0
  367. /package/{core/com → com}/nav/tpl/admin_pc/page_type.vue +0 -0
  368. /package/{core/com → com}/nav/tpl/dev_pc/page_channel.vue +0 -0
  369. /package/{core/com → com}/nav/tpl/dev_pc/page_config.vue +0 -0
  370. /package/{core/com → com}/nav/tpl/dev_pc/page_form.vue +0 -0
  371. /package/{core/com → com}/nav/tpl/dev_pc/page_nav.vue +0 -0
  372. /package/{core/com → com}/nav/tpl/dev_pc/page_table.vue +0 -0
  373. /package/{core/com → com}/nav/tpl/home_pc/page_channel.vue +0 -0
  374. /package/{core/com → com}/nav/tpl/home_phone/page_list.vue +0 -0
  375. /package/{core/com → com}/param/README.md +0 -0
  376. /package/{core/com/cmd → com/pendant}/README.md +0 -0
  377. /package/{core/com → com}/socket/README.md +0 -0
  378. /package/{core/com → com}/sql/README.md +0 -0
  379. /package/{core/com → com}/static/README.md +0 -0
  380. /package/{core/com → com}/task/README.md +0 -0
@@ -0,0 +1,1259 @@
1
+ const Item = require('mm_machine').Drive;
2
+ const Excel = require('mm_excel');
3
+
4
+ /**
5
+ * Sql操作驱动类
6
+ * @augments {Item}
7
+ * @class
8
+ */
9
+ class Drive extends Item {
10
+ static config = {
11
+ 'main': '',
12
+ // 表名 {0} 代表可前端传参自定义查询的表
13
+ 'table': '{0}',
14
+ // 主键 用于水平连表查询时 例如:id
15
+ 'key': '',
16
+ // 排序 {0} 代表可前台传参自定义排序规则 格式: `name` asc, `id` desc
17
+ 'orderby': '{0}',
18
+ // 默认排序 `id` desc
19
+ 'orderby_default': '',
20
+ // 显示的字段 {0} 代表可前台传参自定义查询的字段, 例如: `id`,`username`,`name`,`email`
21
+ 'field': '{0}',
22
+ // 默认显示字段 例如: `id`,`username`,`name`
23
+ 'field_default': '*',
24
+ // 隐藏字段 有些字段即使前端请求也不能返回,这是通过隐藏字段将其过滤掉, 例如:password *表示包含匹配
25
+ 'field_hide': ['*password*', '*token*', 'salt'],
26
+ // 分页大小,默认一页显示条数
27
+ 'page_size': 30,
28
+ /* 过滤参数 */
29
+ 'filter': {
30
+ /**
31
+ * 表名
32
+ */
33
+ 'table': 'table',
34
+ /**
35
+ * 查询的页码
36
+ */
37
+ 'page': 'page',
38
+ /**
39
+ * 查询每页条数
40
+ */
41
+ 'size': 'size',
42
+ /**
43
+ * 操作方式: 传入参数method=add, 支持参数 add增、del删、set改、get查,为空则为get
44
+ */
45
+ 'method': 'method',
46
+ /**
47
+ * 排序
48
+ */
49
+ 'orderby': 'orderby',
50
+ /**
51
+ * 查询显示的字段
52
+ */
53
+ 'field': 'field',
54
+ /**
55
+ * 统计结果: 统计符合条件的结果数,只有当page等于1或0时才会统计
56
+ */
57
+ 'count_ret': 'count_ret'
58
+ },
59
+ // 分隔符 用于查询时的多条件处理
60
+ 'separator': '|',
61
+ // 支持的方法 add增、del删、set改、get查, 只填get表示只支持查询 // import export del_repeat",
62
+ 'method': 'add del set get get_obj import export del_repeat avg sum count update',
63
+ // sql查询语句
64
+ 'query': {},
65
+ // 默认查询, 当查询条件中不包含该项时,默认添加该项。 例如: { "age": "`age` < 20" } , 当查询参含有age,不调用该项,不存在时,sql会增加该项
66
+ 'query_default': {},
67
+ // sql更改语句
68
+ 'update': {},
69
+ // 默认添加条件,当不包含该项时,默认添加该项。 例如: { "age": "`age` += 1" } , 当查询参含有age,不调用该项,不存在时,sql会增加该项
70
+ 'body_default': {},
71
+ // 参数 []
72
+ 'params': null,
73
+ // 格式
74
+ 'format': [
75
+ /* {
76
+ // 表名,当选择转换方式 表转换时需填写
77
+ "table": "mm_web_region",
78
+ // 查询条件,用于加速转换
79
+ "query": {
80
+ "group": "市"
81
+ },
82
+ // 表标题名
83
+ "title": "所属省份",
84
+ // 转换ID
85
+ "id": "province_id",
86
+ // 转换主键
87
+ "key": "province",
88
+ // 取名
89
+ "name": "name",
90
+ // 列表
91
+ "list": [{
92
+ "province_id": 1,
93
+ "name": "广东省"
94
+ },
95
+ {
96
+ "province_id": 2,
97
+ "name": "广西省"
98
+ },
99
+ {
100
+ "province_id": 3,
101
+ "name": "湖南省"
102
+ }
103
+ ]
104
+ },
105
+ {
106
+ "title": "是否可见",
107
+ "key": "show",
108
+ "list": ["否", "是"]
109
+ } */
110
+ ],
111
+ /* 去重 */
112
+ 'del_repeat': {
113
+ // 判断重复的字段,例如字段名 number
114
+ 'groupBy': '',
115
+ // 排序方式 例如: `diJia` ASC
116
+ 'orderBy': ''
117
+ },
118
+ /* 逻辑符 */
119
+ 'logic': {},
120
+ // 输出sql语句
121
+ 'log': false
122
+ };
123
+
124
+
125
+ /**
126
+ * 构造函数
127
+ * @param {object} config 配置参数
128
+ * @param {object} parent 父对象
129
+ * @class
130
+ */
131
+ constructor(config, parent) {
132
+ super({ ...Drive.config, ...config }, parent);
133
+ // 默认启用热更新
134
+ this.mode = 3;
135
+ }
136
+ }
137
+
138
+ /**
139
+ * 获取模板目录
140
+ * @returns {string} 模板目录
141
+ */
142
+ Drive.prototype._getTplDir = function () {
143
+ return __dirname;
144
+ };
145
+
146
+ /**
147
+ * 预设
148
+ */
149
+ Drive.prototype._preset = function () {
150
+ // 保存文件目录
151
+ this.save_dir = './static/file/';
152
+ // 读取文件目录
153
+ this.url_path = '/file/';
154
+ /* 通用项 */
155
+ this.params = null;
156
+ };
157
+
158
+ /**
159
+ * 执行前, 可用于过滤参数
160
+ * @param {object} params 参数对象 (object) 包含query和body 如{ query, body }
161
+ * @param {object} db 数据库管理器
162
+ * @returns {object} 过滤后的参数
163
+ */
164
+ Drive.prototype.before = async function (params, db) {
165
+ return params;
166
+ };
167
+
168
+ /**
169
+ * 验证, 用于判断是否执行
170
+ * @param {object} params 参数对象 (object) 包含query和body
171
+ * @param {object} db 数据管理器
172
+ * @returns {boolean} 验证通过返回true, 失败返回false
173
+ */
174
+ Drive.prototype.check = async function (params, db) {
175
+ return true;
176
+ };
177
+
178
+ /**
179
+ * 执行后, 可用于附加执行
180
+ * @param {object} params 参数对象 (object) 包含query和body
181
+ * @param {object} db 数据管理器
182
+ * @returns {object} 最终执行结果
183
+ */
184
+ Drive.prototype.after = async function (params, db) {
185
+ return db.ret;
186
+ };
187
+
188
+ /**
189
+ * 更新缓存
190
+ * @param {string} table 表名
191
+ * @param {object} sql 缓存sql语句
192
+ */
193
+ Drive.prototype.updateCache = async function (table, sql) {
194
+
195
+ };
196
+
197
+ /**
198
+ * @ 执行修改
199
+ * @param {object} query 查询url参数
200
+ * @param {object} body 修改时置入body的参数
201
+ * @param {object} db 数据库管理器
202
+ * @returns {object} 返回执行结果
203
+ */
204
+ Drive.prototype.run = async function (query, body, db) {
205
+ var params = {
206
+ query,
207
+ body
208
+ };
209
+ params = await this.before(params, db);
210
+ if (this.check(params, db)) {
211
+ var ret = await this.main(params, db);
212
+ if (!ret.error) {
213
+ db.ret = ret;
214
+ return await this.after(params, db);
215
+ } else {
216
+ return ret;
217
+ }
218
+ }
219
+ return null;
220
+ };
221
+
222
+
223
+ /**
224
+ * SQL操作准备
225
+ * @param {object} db 数据库操作类
226
+ * @param {object} query 查询条件
227
+ * @param {object} body 修改项
228
+ * @returns {object} 返回准备参数
229
+ */
230
+ Drive.prototype.ready = function (db, query, body) {
231
+ var cg = this.config;
232
+ var qy = { ...query };
233
+ $.push(db.config.filter, cg.filter, true);
234
+ db.filter(qy);
235
+ return {
236
+ cg,
237
+ qy
238
+ };
239
+ };
240
+
241
+ /**
242
+ * 转为查询条件字符串
243
+ * @param {object} db 数据库管理器
244
+ * @param {object} query 查寻条件
245
+ * @param {object} method 方法
246
+ * @returns {string} 返回查询条件
247
+ */
248
+ Drive.prototype.toWhere = function (db, query, method) {
249
+ var { cg, qy: originalQy } = this.ready(db, query, {});
250
+ // 创建 qy 对象的副本,避免修改原始参数
251
+ var qy = { ...originalQy };
252
+ this._setDbConfig(db, cg, query);
253
+ qy = this._setQueryField(db, query, cg, method, qy);
254
+ qy = this._setOrderBy(db, query, cg, qy);
255
+ return this._buildQueryString(db, qy, cg);
256
+ };
257
+
258
+ /**
259
+ * 设置数据库配置
260
+ * @param {object} db 数据库管理器
261
+ * @param {object} cg 配置
262
+ * @param {object} query 查询条件
263
+ * @private
264
+ */
265
+ Drive.prototype._setDbConfig = function (db, cg, query) {
266
+ db.config.separator = cg.separator;
267
+ if (!query.size && cg.page_size) {
268
+ db.size = cg.page_size + 0;
269
+ }
270
+ if (db.size > 0 && db.page === 0) {
271
+ db.page = 1;
272
+ }
273
+ };
274
+
275
+ /**
276
+ * 设置查询字段
277
+ * @param {object} db 数据库管理器
278
+ * @param {object} query 查询条件
279
+ * @param {object} cg 配置
280
+ * @param {object} method 方法
281
+ * @param {object} qy 查询条件对象
282
+ * @returns {object} 更新后的查询条件对象
283
+ * @private
284
+ */
285
+ Drive.prototype._setQueryField = function (db, query, cg, method, qy) {
286
+ var f = db.config.filter;
287
+ var field = query[f.field];
288
+ if (cg.field.has('*{0}*')) {
289
+ if (field) {
290
+ db.field = cg.field.replace('{0}', field);
291
+ } else if (method === 'get_obj' && cg.field_obj) {
292
+ db.field = cg.field_obj + '';
293
+ } else if (cg.field_default) {
294
+ db.field = cg.field_default + '';
295
+ }
296
+ } else {
297
+ db.field = cg.field + '';
298
+ if (field) {
299
+ return { ...qy, [f.field]: field };
300
+ }
301
+ }
302
+ return qy;
303
+ };
304
+
305
+ /**
306
+ * 设置排序方式
307
+ * @param {object} db 数据库管理器
308
+ * @param {object} query 查询条件
309
+ * @param {object} cg 配置
310
+ * @param {object} qy 查询条件对象
311
+ * @returns {object} 更新后的查询条件对象
312
+ * @private
313
+ */
314
+ Drive.prototype._setOrderBy = function (db, query, cg, qy) {
315
+ var f = db.config.filter;
316
+ var orderby = query[f.orderby];
317
+ if (cg.orderby.has('*{0}*')) {
318
+ if (orderby) {
319
+ db.orderby = cg.orderby.replace('{0}', orderby);
320
+ } else if (cg.orderby_default) {
321
+ db.orderby = cg.orderby_default + '';
322
+ }
323
+ } else {
324
+ db.orderby = cg.orderby + '';
325
+ if (orderby) {
326
+ return { ...qy, [f.orderby]: orderby };
327
+ }
328
+ }
329
+ return qy;
330
+ };
331
+
332
+ /**
333
+ * 构建查询字符串
334
+ * @param {object} db 数据库管理器
335
+ * @param {object} qy 查询条件对象
336
+ * @param {object} cg 配置
337
+ * @returns {string} 查询字符串
338
+ * @private
339
+ */
340
+ Drive.prototype._buildQueryString = function (db, qy, cg) {
341
+ var query_str = db.tplQuery(qy, cg.query);
342
+ var qt = cg.query_default;
343
+ if (Object.keys(qt).length > 0) {
344
+ var id = $.dict.user_id;
345
+ var id_key = '{' + id + '}';
346
+ var user_id = '0';
347
+ var user = db.user;
348
+ if (user && user[id]) {
349
+ user_id = user[id];
350
+ }
351
+ for (var k in qt) {
352
+ if (!qy[k]) {
353
+ query_str += ' && ' + qt[k].replaceAll(id_key, user_id);
354
+ }
355
+ }
356
+ if (query_str.startsWith(' && ')) {
357
+ query_str = query_str.replace(' && ', '');
358
+ }
359
+ }
360
+ return query_str;
361
+ };
362
+
363
+ /**
364
+ * 查询(主要)
365
+ * @param {object} db 数据库操作类
366
+ * @param {object} query 查询条件
367
+ * @param {object} method 方法
368
+ * @returns {object} 返回查询结果
369
+ */
370
+ Drive.prototype.getMain = async function (db, query, method) {
371
+ var ret;
372
+ if (this.config.field.has('*{0}*')) {
373
+ var field = query[db.config.filter.field];
374
+ if (field && this.config.field_hide.getMatch(field)) {
375
+ return $.ret.error(70003, '不合法的查询参数');
376
+ }
377
+ }
378
+ var query_str = this.toWhere(db, query, method);
379
+
380
+ // 查询
381
+ if (db.count_ret === 'true') {
382
+ ret = $.ret.body(await db.getCountSql(query_str, db.orderby, db.field));
383
+ } else {
384
+ ret = $.ret.list(await db.getSql(query_str, db.orderby, db.field));
385
+ }
386
+ return ret;
387
+ };
388
+
389
+ /**
390
+ * 查询
391
+ * @param {object} db 数据库操作类
392
+ * @param {object} query 查询条件
393
+ * @param {object} body 修改项
394
+ * @returns {object} 返回查询结果
395
+ */
396
+ Drive.prototype.get = async function (db, query, body) {
397
+ return await this.getMain(db, query);
398
+ };
399
+
400
+ /**
401
+ * 查询单条数据
402
+ * @param {object} db 数据库操作类
403
+ * @param {object} query 查询条件
404
+ * @param {object} body 修改项
405
+ * @returns {object} 返回查询结果
406
+ */
407
+ Drive.prototype.getObj = async function (db, query, body) {
408
+ var qy = { ...query };
409
+ qy.page = 1;
410
+ qy.size = 1;
411
+ var ret = await this.getMain(db, qy, 'get_obj');
412
+ if (ret.result) {
413
+ var list = ret.result.list;
414
+ if (list.length > 0) {
415
+ return $.ret.obj(list[0]);
416
+ } else {
417
+ return $.ret.obj(null);
418
+ }
419
+ }
420
+ return ret;
421
+ };
422
+
423
+ /**
424
+ * 修改(主要)
425
+ * @param {object} db 数据库操作类
426
+ * @param {object} query 查询条件
427
+ * @param {object} body 修改项
428
+ * @returns {object} 返回修改结果
429
+ */
430
+ Drive.prototype.setMain = async function (db, query, body) {
431
+ var ret;
432
+ var {
433
+ cg,
434
+ qy
435
+ } = this.ready(db, query, body);
436
+ var key = cg.key;
437
+ if (body[key]) {
438
+ qy[key] = body[key];
439
+ }
440
+ var query_str = db.tplQuery(qy, cg.query);
441
+
442
+ var qt = cg.query_default;
443
+ if (Object.keys(qt).length > 0) {
444
+ var id = $.dict.user_id;
445
+ var id_key = '{' + id + '}';
446
+ var user_id = '0';
447
+ var user = db.user;
448
+ if (user && user[id]) {
449
+ user_id = user[id];
450
+ }
451
+ for (var k in qt) {
452
+ if (!qy[k]) {
453
+ query_str += ' && ' + qt[k].replace(id_key, user_id);
454
+ }
455
+ }
456
+ if (query_str.startsWith(' && ')) {
457
+ query_str = query_str.replace(' && ', '');
458
+ }
459
+ }
460
+
461
+ var n = await db.setSql(query_str, db.tplBody(body, cg.update));
462
+
463
+ if (n < 1) {
464
+ ret = $.ret.error(500, '修改失败!\n' + db.error.message);
465
+ } else {
466
+ ret = $.ret.bl(true, '修改成功!');
467
+ }
468
+ return ret;
469
+ };
470
+
471
+
472
+ /**
473
+ * 修改
474
+ * @param {object} db 数据库操作类
475
+ * @param {object} query 查询条件
476
+ * @param {object} body 修改项
477
+ * @returns {object} 返回查询结果
478
+ */
479
+ Drive.prototype.set = async function (db, query, body) {
480
+ return await this.setMain(db, query, body);
481
+ };
482
+
483
+ /**
484
+ * 添加(主要)
485
+ * @param {object} db 数据库操作类
486
+ * @param {object} body 添加项
487
+ * @returns {object} 返回查询结果
488
+ */
489
+ Drive.prototype.addMain = async function (db, body) {
490
+ var ret;
491
+ var cg = this.config;
492
+ if (Object.keys(body).length > 0) {
493
+ var bt = cg.body_default;
494
+ if (Object.keys(bt).length > 0) {
495
+ var id = $.dict.user_id;
496
+ var id_key = '{' + id + '}';
497
+ var user_id = '0';
498
+ var user = db.user;
499
+ if (user && user[id]) {
500
+ user_id = user[id];
501
+ }
502
+ for (var k in bt) {
503
+ if (!body[k]) {
504
+ var str = bt[k].replace(id_key, user_id);
505
+ var key = str.left('=').trim('`');
506
+ body[key] = str.right('=').trim("'");
507
+ }
508
+ }
509
+ }
510
+ var n = await db.add(body);
511
+ if (n < 1) {
512
+ ret = $.ret.error(500, '添加失败!\n' + db.error.message);
513
+ } else {
514
+ ret = $.ret.bl(true, '添加成功!');
515
+ }
516
+ } else {
517
+ ret = $.ret.error(30001, '参数不能为空');
518
+ }
519
+ if (cg.log) {
520
+ this.log('debug', '添加SQL语句', db.sql);
521
+ }
522
+ return ret;
523
+ };
524
+
525
+ /**
526
+ * 添加
527
+ * @param {object} db 数据库操作类
528
+ * @param {object} query 查询条件
529
+ * @param {object} body 修改项
530
+ * @returns {object} 返回添加结果
531
+ */
532
+ Drive.prototype.add = async function (db, query, body) {
533
+ return await this.addMain(db, body);
534
+ };
535
+
536
+ /**
537
+ * 删除(主要)
538
+ * @param {object} db 数据库操作类
539
+ * @param {object} query 查询条件
540
+ * @returns {object} 返回查询结果
541
+ */
542
+ Drive.prototype.delMain = async function (db, query) {
543
+ var ret;
544
+ var {
545
+ cg,
546
+ qy
547
+ } = this.ready(db, query, {});
548
+ var query_str = db.tplQuery(qy, cg.query);
549
+ var bl = await db.delSql(query_str);
550
+ if (bl < 1) {
551
+ ret = $.ret.error(500, '删除失败!\n' + db.error.message);
552
+ } else {
553
+ ret = $.ret.bl(true, '删除成功!');
554
+ }
555
+ if (cg.log) {
556
+ this.log('debug', '删除SQL语句', db.sql);
557
+ }
558
+ return ret;
559
+ };
560
+
561
+ /**
562
+ * 删除
563
+ * @param {object} db 数据库操作类
564
+ * @param {object} query 查询条件
565
+ * @param {object} body 修改项
566
+ * @returns {object} 返回删除结果
567
+ */
568
+ Drive.prototype.del = async function (db, query, body = {}) {
569
+ return await this.delMain(db, { ...query, ...body });
570
+ };
571
+
572
+ /**
573
+ * 添加或修改(主要)
574
+ * @param {object} db 数据库操作类
575
+ * @param {object} query 查询条件
576
+ * @param {object} body 修改项
577
+ * @returns {object} 返回修改结果
578
+ */
579
+ Drive.prototype.addorsetMain = async function (db, query, body) {
580
+ var ret;
581
+ var {
582
+ cg,
583
+ qy
584
+ } = this.ready(db, query, body);
585
+ if (Object.keys(body).length > 0 && Object.keys(qy).length > 0) {
586
+ var n = await db.addOrSet(qy, body);
587
+ if (n < 1) {
588
+ ret = $.ret.error(500, '操作失败!\n' + db.error.message);
589
+ } else {
590
+ ret = $.ret.bl(true, '操作成功!');
591
+ }
592
+ } else {
593
+ ret = $.ret.error(30001, '参数不能为空');
594
+ }
595
+ if (cg.log) {
596
+ this.log('debug', '添加或修改SQL语句', db.sql);
597
+ }
598
+ return ret;
599
+ };
600
+
601
+ /**
602
+ * 添加或修改
603
+ * @param {object} db 数据库操作类
604
+ * @param {object} query 查询条件
605
+ * @param {object} body 修改项
606
+ * @returns {object} 返回查询结果
607
+ */
608
+ Drive.prototype.addOrSet = async function (db, query, body) {
609
+ return await this.addOrSetMain(db, query, body);
610
+ };
611
+
612
+ /**
613
+ * 获取数据类型
614
+ * @param {string} data_type 数据类型
615
+ * @returns {string} 返回数据类型
616
+ */
617
+ Drive.prototype.getType = function (data_type) {
618
+ if (data_type.indexOf('int') !== -1) {
619
+ return 'number';
620
+ } else {
621
+ return 'string';
622
+ }
623
+ };
624
+
625
+ /**
626
+ * 获取数据
627
+ * @param {object} db 数据库操作类
628
+ * @param {string} field 要获取的字段
629
+ * @returns {Array} 返回参数信息列表
630
+ */
631
+ Drive.prototype.getParam = async function (db, field) {
632
+ var cg = this.config;
633
+ if (cg.params) {
634
+ return cg.params;
635
+ }
636
+ if (this.params) {
637
+ return this.params;
638
+ }
639
+ var lt = await this._getFieldsFromCache(cg);
640
+ if (!lt.length) {
641
+ lt = await this._getFieldsFromDb(db, cg);
642
+ }
643
+ return this._filterFields(lt, field, cg);
644
+ };
645
+
646
+ /**
647
+ * 从缓存获取字段列表
648
+ * @param {object} cg 配置
649
+ * @returns {Array} 字段列表
650
+ * @private
651
+ */
652
+ Drive.prototype._getFieldsFromCache = function (cg) {
653
+ var lt = [];
654
+ if ($.pool.db) {
655
+ var pool = $.pool.db['sys'];
656
+ if (pool) {
657
+ var dt = pool.get(cg.table);
658
+ if (dt && dt.config) {
659
+ var list = dt.config.fields;
660
+ for (var i = 0; i < list.length; i++) {
661
+ var name = list[i].replace(/`/g, '');
662
+ lt.push({ name, title: name });
663
+ }
664
+ }
665
+ }
666
+ }
667
+ return lt;
668
+ };
669
+
670
+ /**
671
+ * 从数据库获取字段列表
672
+ * @param {object} db 数据库操作类
673
+ * @param {object} cg 配置
674
+ * @returns {Array} 字段列表
675
+ * @private
676
+ */
677
+ Drive.prototype._getFieldsFromDb = async function (db, cg) {
678
+ var lt = [];
679
+ db.table = cg.table;
680
+ var list = await db.fields();
681
+ for (var i = 0; i < list.length; i++) {
682
+ var o = list[i];
683
+ var name = o.name;
684
+ var note = o.note ? o.note.replace(':', ':') : name;
685
+ lt.push({
686
+ name,
687
+ title: note.left(':', true),
688
+ type: this.getType(o.type)
689
+ });
690
+ }
691
+ return lt;
692
+ };
693
+
694
+ /**
695
+ * 过滤字段
696
+ * @param {Array} lt 字段列表
697
+ * @param {string} field 字段筛选
698
+ * @param {object} cg 配置
699
+ * @returns {Array} 过滤后的字段列表
700
+ * @private
701
+ */
702
+ Drive.prototype._filterFields = function (lt, field, cg) {
703
+ var fields = field || cg.field_default;
704
+ if (fields && fields !== '*') {
705
+ var arr = fields.replace(/`/g, '').split(',');
706
+ return arr.map(name => lt.getObj({ name })).filter(obj => obj);
707
+ }
708
+ return lt;
709
+ };
710
+
711
+ /**
712
+ * 获取导入导出格式
713
+ * @param {object} db 数据库操作类
714
+ * @returns {Array} 格式列表
715
+ */
716
+ Drive.prototype.getFormat = async function (db) {
717
+ var dbs = { ...db };
718
+ dbs.size = 0;
719
+ var fmt = this.config.format;
720
+ for (var i = 0; i < fmt.length; i++) {
721
+ var o = fmt[i];
722
+ if (o.table) {
723
+ if (!o.list || o.list.length == 0) {
724
+ dbs.table = o.table;
725
+ if (!o.id) {
726
+ o.id = o.key;
727
+ }
728
+ o.list = await dbs.getSql(o.where, null, o.id + ',' + o.name);
729
+ if (o.id !== o.key) {
730
+ for (var j = 0; j < o.list.length; j++) {
731
+ var m = o.list[j];
732
+ m[o.key] = m[o.name];
733
+ }
734
+ }
735
+ }
736
+ }
737
+ }
738
+ return fmt;
739
+ };
740
+
741
+ /**
742
+ * 导入数据(主要)
743
+ * @param {object} db 数据库操作类
744
+ * @param {string} filename 文件名
745
+ * @returns {object} 返回导入结果
746
+ */
747
+ Drive.prototype.importMain = async function (db, filename) {
748
+ var params = await this.getParam(db);
749
+ var format = await this.getFormat(db);
750
+ var file = this._processFilePath(filename);
751
+ if (!file.hasFile()) {
752
+ return $.ret.error(30001, file + '文件不存在!');
753
+ }
754
+ var jarr = await this._loadImport(file, params, format);
755
+ if (!jarr || !jarr.length) {
756
+ return $.ret.error(10000, '要导入的数据不能为空!');
757
+ }
758
+ var result = await this._importToDb(db, jarr);
759
+ return this._buildImport(result);
760
+ };
761
+
762
+ /**
763
+ * 处理文件路径
764
+ * @param {string} filename 文件名
765
+ * @returns {string} 处理后的文件路径
766
+ * @private
767
+ */
768
+ Drive.prototype._processFilePath = function (filename) {
769
+ var file = filename;
770
+ if (file.indexOf($.run_path) !== 0) {
771
+ file = file.replace(this.url_path, this.save_dir);
772
+ var path = this._getPath();
773
+ file = file.fullname(path);
774
+ }
775
+ return file;
776
+ };
777
+
778
+ /**
779
+ * 加载导入数据
780
+ * @param {string} file 文件路径
781
+ * @param {Array} params 参数列表
782
+ * @param {Array} format 格式列表
783
+ * @returns {Array} 数据数组
784
+ * @private
785
+ */
786
+ Drive.prototype._loadImport = async function (file, params, format) {
787
+ if (file.ends('.json')) {
788
+ return file.loadJson();
789
+ } else if (file.ends('.xml')) {
790
+ return file.loadXml();
791
+ } else {
792
+ var excel = new Excel({ file, params, format });
793
+ try {
794
+ return await excel.load();
795
+ } catch (e) {
796
+ this.log('error', '导入文件', e);
797
+ } finally {
798
+ excel.clear();
799
+ excel = null;
800
+ }
801
+ }
802
+ return [];
803
+ };
804
+
805
+ /**
806
+ * 导入数据到数据库
807
+ * @param {object} db 数据库操作类
808
+ * @param {Array} jarr 数据数组
809
+ * @returns {object} 导入结果
810
+ * @private
811
+ */
812
+ Drive.prototype._importToDb = async function (db, jarr) {
813
+ var list = [];
814
+ var errors = [];
815
+ var list_error = [];
816
+ db.table = db.table || this.config.table;
817
+ var key = this.config.key;
818
+ if (jarr[0][key]) {
819
+ for (var i = 0; i < jarr.length; i++) {
820
+ var o = jarr[i];
821
+ var qy = {};
822
+ qy[key] = o[key];
823
+ var n = await db.addOrSet(qy, o);
824
+ if (n < 1) {
825
+ errors.push(db.error);
826
+ list_error.push(o);
827
+ } else {
828
+ list.push(o);
829
+ }
830
+ }
831
+ } else {
832
+ for (var i = 0; i < jarr.length; i++) {
833
+ var o = jarr[i];
834
+ var n = await db.add(o);
835
+ if (n < 1) {
836
+ errors.push(db.error);
837
+ list_error.push(o);
838
+ } else {
839
+ list.push(o);
840
+ }
841
+ }
842
+ }
843
+ return { list, errors, list_error, total: jarr.length };
844
+ };
845
+
846
+ /**
847
+ * 构建导入结果
848
+ * @param {object} result 导入结果数据
849
+ * @returns {object} 返回结果对象
850
+ * @private
851
+ */
852
+ Drive.prototype._buildImport = function (result) {
853
+ var bl = result.list.length === result.total;
854
+ var body = $.ret.bl(bl, bl ? '导入成功!' : '导入失败!');
855
+ body.result.list = result.list;
856
+ if (result.errors.length) {
857
+ body.result.list_error = result.list_error;
858
+ body.result.errors = result.errors;
859
+ }
860
+ return body;
861
+ };
862
+
863
+
864
+ /**
865
+ * 导入数据
866
+ * @param {object} db 数据库操作类
867
+ * @param {object} query 查询条件
868
+ * @param {object} body 导入设置
869
+ * @returns {object} 返回执行结果
870
+ */
871
+ Drive.prototype.import = async function (db, query, body) {
872
+ var params = { ...query, ...body };
873
+ if (!params.file) {
874
+ return $.ret.error(60000, '文件名(file)参数不能为空');
875
+ }
876
+ return await this.importMain(db, params.file);
877
+ };
878
+
879
+ /**
880
+ * 导出数据(主要)
881
+ * @param {object} db 数据库操作类
882
+ * @param {object} query 查询参数
883
+ * @param {object} body 正文参数(导出设置)
884
+ * @property {string} body.fields 需要导出的字段 例如: `username`,`gm`,`vip`
885
+ * @property {string} body.file 文件名 例如: 用户名.xlsx 、用户信息.csv 、用户账户.xls
886
+ * @property {string} body.path 文件路径 例如: /static/download, 可不填写
887
+ * @returns {object} 返回执行结果
888
+ */
889
+ Drive.prototype.exportMain = async function (db, query, body) {
890
+ var by = await this.getMain(db, query);
891
+ if (db.error) {
892
+ return $.ret.error(10000, db.error.message);
893
+ }
894
+ var config = this._prepareExportConfig(query, body);
895
+ var params = await this.getParam(db, config.fields);
896
+ var format = await this.getFormat(db);
897
+ var file = await this._saveExportFile(config.file, params, format, by.result.list);
898
+ return this._buildExport(file, config.name);
899
+ };
900
+
901
+ /**
902
+ * 准备导出配置
903
+ * @param {object} query 查询参数
904
+ * @param {object} body 正文参数
905
+ * @returns {object} 导出配置
906
+ * @private
907
+ */
908
+ Drive.prototype._prepareExportConfig = function (query, body) {
909
+ var path = body.path;
910
+ var file = body.file;
911
+ var fields = body.fields;
912
+ var table = $.dict.table || this.config.table;
913
+ if (!path) {
914
+ path = this._getPath();
915
+ }
916
+ if (!file) {
917
+ var date = new Date();
918
+ file = this.save_dir + table + '_' + date.stamp() + '.xlsx';
919
+ }
920
+ if (!fields) {
921
+ fields = query.field || (this.config.field_default !== '*' ? this.config.field_default : '');
922
+ }
923
+ return { path, file, fields, name: file.split('/').pop() };
924
+ };
925
+
926
+ /**
927
+ * 获取路径
928
+ * @returns {string} 导出路径
929
+ * @private
930
+ */
931
+ Drive.prototype._getPath = function () {
932
+ return $.config.get('user_path') || $.config.get('static_path') || './static/';
933
+ };
934
+
935
+ /**
936
+ * 保存导出文件
937
+ * @param {string} file 文件路径
938
+ * @param {Array} params 参数列表
939
+ * @param {Array} format 格式列表
940
+ * @param {Array} list 数据列表
941
+ * @returns {string} 保存后的文件路径
942
+ * @private
943
+ */
944
+ Drive.prototype._saveExportFile = async function (file, params, format, list) {
945
+ var export_path = this._getPath();
946
+ var save_file = file.fullname(export_path);
947
+ save_file.addDir();
948
+ var excel = new Excel({ file: save_file, params, format });
949
+ try {
950
+ return await excel.save(list);
951
+ } catch (e) {
952
+ this.log('error', '导出保存文件失败!', e);
953
+ } finally {
954
+ excel.clear();
955
+ excel = null;
956
+ }
957
+ return '';
958
+ };
959
+
960
+ /**
961
+ * 构建导出结果
962
+ * @param {string} file 文件路径
963
+ * @param {string} name 文件名
964
+ * @returns {object} 导出结果
965
+ * @private
966
+ */
967
+ Drive.prototype._buildExport = function (file, name) {
968
+ var body = $.ret.bl(!!file, file ? '导出成功!' : '导出失败!');
969
+ body.result.file = file;
970
+ body.result.url = this.url_path + name;
971
+ return body;
972
+ };
973
+
974
+ /**
975
+ * 导出数据
976
+ * @param {object} db 数据库操作类
977
+ * @param {object} query 查询条件
978
+ * @param {object} body 修改项
979
+ * @returns {object} 返回执行结果
980
+ */
981
+ Drive.prototype.export = async function (db, query, body) {
982
+ return await this.exportMain(db, query, body);
983
+ };
984
+
985
+ /**
986
+ * 删除重复项(主要)
987
+ * @param {object} db 数据库管理器
988
+ * @param {object} params 查询参数
989
+ * @returns {object} 返回执行结果
990
+ */
991
+ Drive.prototype.delRepeatMain = async function (db, params) {
992
+ var pm = { ...params };
993
+ var msg = '';
994
+ var cg = this.config.del_repeat;
995
+ var order_by = pm.orderby || cg.orderBy;
996
+ delete pm.orderby;
997
+ var group_by = pm.groupby || cg.groupBy;
998
+ delete pm.groupby;
999
+ var f = db.config.filter;
1000
+ // 设置查询字段
1001
+ var field = pm[f.field] || group_by;
1002
+ delete pm[f.field];
1003
+ db.field = field;
1004
+ var sql = db.toGetSql(pm).replace(' * ', ' `' + field + '`, count(*) as len ');
1005
+ sql += ` GROUP BY ${group_by}`;
1006
+ sql = 'SELECT * FROM (' + sql + ') a WHERE len > 1';
1007
+ var list = await db.run(sql);
1008
+ if (list.length) {
1009
+ db.page = 1;
1010
+ db.size = 1;
1011
+ var key = this.config.key;
1012
+ for (var i = 0; i < list.length; i++) {
1013
+ var o = list[i];
1014
+ var len = o.len - 1;
1015
+ delete o.len;
1016
+ for (var n = 0; n < len; n++) {
1017
+ var obj = await db.getObj(o, order_by, key);
1018
+ if (obj) {
1019
+ await db.del(obj);
1020
+ }
1021
+ }
1022
+ }
1023
+ } else {
1024
+ msg = '没有重复项。';
1025
+ }
1026
+ return $.ret.bl(!msg, msg ? '去重失败!原因:' + msg : '去重成功!');
1027
+ };
1028
+
1029
+ /**
1030
+ * 删除重复项
1031
+ * @param {object} db 数据库管理器
1032
+ * @param {object} query 查询条件
1033
+ * @param {object} body 正文参数
1034
+ * @returns {object} 返回执行结果
1035
+ */
1036
+ Drive.prototype.delRepeat = async function (db, query, body) {
1037
+ var pm = { ...query, ...body };
1038
+ return await this.delRepeatMain(db, pm);
1039
+ };
1040
+
1041
+ /**
1042
+ * 执行模板操作
1043
+ * @param {object} params 参数对象 (object) 包含query和body
1044
+ * @param {object} db 数据管理器
1045
+ * @returns {object} 返回执行结果
1046
+ */
1047
+ Drive.prototype.main = async function (params, db) {
1048
+ var {
1049
+ query,
1050
+ body
1051
+ } = params;
1052
+
1053
+ var cg = this.config;
1054
+ var method = query.method;
1055
+
1056
+ if (!method) {
1057
+ method = 'get';
1058
+ }
1059
+ if (!cg.method.has('*' + method + '*')) {
1060
+ return $.ret.error(50001, '不支持的操作方式');
1061
+ }
1062
+ method = this._getMethodName(method);
1063
+ var qy = { ...query };
1064
+ delete qy.method;
1065
+ if (this[method]) {
1066
+ db.method = method;
1067
+
1068
+ // 过滤查询参数
1069
+ var f = cg.filter;
1070
+ var table = query[f.table];
1071
+
1072
+ // 设置操作的数据表
1073
+ if (cg.table.has('*{0}*')) {
1074
+ if (table) {
1075
+ db.table = cg.table.replace('{0}', table);
1076
+ } else {
1077
+ return $.ret.error(30001, '表名不能为空');
1078
+ }
1079
+ } else {
1080
+ db.table = cg.table + '';
1081
+ }
1082
+
1083
+ return await this[method](db, qy, { ...body });
1084
+ } else {
1085
+ return $.ret.error(50001, '不支持的操作方式');
1086
+ }
1087
+ };
1088
+
1089
+ /**
1090
+ * 获取方法名
1091
+ * @param {string} method 方法名
1092
+ * @returns {string} 方法名
1093
+ * @private
1094
+ */
1095
+ Drive.prototype._getMethodName = function (method) {
1096
+ if (!method) {
1097
+ return '';
1098
+ }
1099
+ if (method.indexOf('_') == -1) {
1100
+ return method;
1101
+ }
1102
+ // 将小写蛇形改为小驼峰
1103
+ let arr = method.split('_');
1104
+ let new_name = arr[0].toLowerCase();
1105
+ for (let i = 1; i < arr.length; i++) {
1106
+ let name = arr[i];
1107
+ new_name += name.charAt(0).toUpperCase() + name.substring(1);
1108
+ }
1109
+ return new_name;
1110
+ };
1111
+
1112
+ /**
1113
+ * 总计
1114
+ * @param {object} db 数据库管理器
1115
+ * @param {object} param 查询条件
1116
+ * @returns {object} 返回执行结果
1117
+ */
1118
+ Drive.prototype.sumMain = async function (db, param) {
1119
+ var pm = { ...param };
1120
+ var ret;
1121
+ var orderby = pm.orderby || '';
1122
+ delete pm.orderby;
1123
+ var groupby = pm.groupby;
1124
+ delete pm.groupby;
1125
+ var f = db.config.filter;
1126
+ var field = pm[f.field];
1127
+ delete pm[f.field];
1128
+ var query_str = this.toWhere(db, pm, 'get_list');
1129
+
1130
+ if (!groupby || !field) {
1131
+ ret = $.ret.error(30000, '参数groupby、field是必须的,且值不能为空!');
1132
+ } else {
1133
+ var list = await db.groupSumSql(query_str, groupby, field, orderby);
1134
+ if (!list.length && db.error) {
1135
+ ret = $.ret.body(db.error);
1136
+ } else {
1137
+ ret = $.ret.list(list);
1138
+ }
1139
+ }
1140
+ return ret;
1141
+ };
1142
+
1143
+ /**
1144
+ * 总计
1145
+ * @param {object} db 数据库管理器
1146
+ * @param {object} query 查询条件
1147
+ * @param {object} body 正文参数
1148
+ * @returns {object} 返回执行结果
1149
+ */
1150
+ Drive.prototype.sum = async function (db, query, body) {
1151
+ var pm = { ...query, ...body };
1152
+ return await this.sumMain(db, pm);
1153
+ };
1154
+
1155
+ /**
1156
+ * 平均值
1157
+ * @param {object} db 数据库管理器
1158
+ * @param {object} param 查询条件
1159
+ * @returns {object} 返回执行结果
1160
+ */
1161
+ Drive.prototype.avgMain = async function (db, param) {
1162
+ var pm = { ...param };
1163
+ var ret;
1164
+ var orderby = pm.orderby || '';
1165
+ delete pm.orderby;
1166
+ var groupby = pm.groupby;
1167
+ delete pm.groupby;
1168
+ var f = db.config.filter;
1169
+ var field = pm[f.field];
1170
+ delete pm[f.field];
1171
+ var query_str = this.toWhere(db, pm, 'get_list');
1172
+
1173
+ if (!groupby || !field) {
1174
+ ret = $.ret.error(30000, '参数groupby、field是必须的,且值不能为空!');
1175
+ } else {
1176
+ var list = await db.groupAvgSql(query_str, groupby, field, orderby);
1177
+ if (!list.length && db.error) {
1178
+ ret = $.ret.body(db.error);
1179
+ } else {
1180
+ ret = $.ret.list(list);
1181
+ }
1182
+ }
1183
+ return ret;
1184
+ };
1185
+
1186
+ /**
1187
+ * 平均值
1188
+ * @param {object} db 数据库管理器
1189
+ * @param {object} query 查询条件
1190
+ * @param {object} body 正文参数
1191
+ * @returns {object} 返回执行结果
1192
+ */
1193
+ Drive.prototype.avg = async function (db, query, body) {
1194
+ var pm = { ...query, ...body };
1195
+ return await this.avgMain(db, pm);
1196
+ };
1197
+
1198
+ /**
1199
+ * 总计
1200
+ * @param {object} db 数据库管理器
1201
+ * @param {object} param 查询条件
1202
+ * @returns {object} 返回执行结果
1203
+ */
1204
+ Drive.prototype.countMain = async function (db, param) {
1205
+ var pm = { ...param };
1206
+ var ret;
1207
+ var orderby = pm.orderby || '';
1208
+ delete pm.orderby;
1209
+ var groupby = pm.groupby;
1210
+ delete pm.groupby;
1211
+ var f = db.config.filter;
1212
+ var field = pm[f.field];
1213
+ delete pm[f.field];
1214
+ var query_str = this.toWhere(db, pm, 'get_list');
1215
+
1216
+ if (!groupby || !field) {
1217
+ ret = $.ret.error(30000, '参数groupby、field是必须的,且值不能为空!');
1218
+ } else {
1219
+ var list = await db.groupCountSql(query_str, groupby, field, orderby);
1220
+ if (!list.length && db.error) {
1221
+ ret = $.ret.body(db.error);
1222
+ } else {
1223
+ ret = $.ret.list(list);
1224
+ }
1225
+ }
1226
+ return ret;
1227
+ };
1228
+
1229
+ /**
1230
+ * 总计
1231
+ * @param {object} db 数据库管理器
1232
+ * @param {object} query 查询条件
1233
+ * @param {object} body 正文参数
1234
+ * @returns {object} 返回执行结果
1235
+ */
1236
+ Drive.prototype.count = async function (db, query, body) {
1237
+ var pm = { ...query, ...body };
1238
+ return await this.countMain(db, pm);
1239
+ };
1240
+
1241
+
1242
+ /**
1243
+ * 获取模型
1244
+ * @param {string} type 模型类型
1245
+ * @returns {object} 返回获取到的模型
1246
+ */
1247
+ Drive.prototype.getModel = function (type) {
1248
+ let model = { ...this.config };
1249
+ let dir = this._getDir();
1250
+ let l = $.slash;
1251
+ let app_name = dir.between('app' + l, l);
1252
+ let plugin_name = dir.between('plugin' + l, l);
1253
+ let name = dir.basename();
1254
+ model.app = app_name;
1255
+ model.plugin = plugin_name;
1256
+ model.name = model.name || app_name + '_' + name;
1257
+ return model;
1258
+ };
1259
+ module.exports = Drive;