user.router.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442
  1. const Router = require('@koa/router');
  2. const router = new Router({ prefix: '/api/user' });
  3. const util = require('../utils/util');
  4. const userService = require('../service/user.service');
  5. const projectService = require('../service/projects.service');
  6. const projectUserService = require('../service/project.user.service');
  7. const pageRoleService = require('../service/pagesRole.service');
  8. const pageService = require('../service/pages.service');
  9. const menuService = require('../service/menu.service');
  10. const publishService = require('../service/publish.service');
  11. const roleService = require('../service/roles.service');
  12. const imgcloudService = require('../service/imgcloud.service');
  13. const md5 = require('md5.js');
  14. const nodemailer = require('nodemailer');
  15. const config = require('../config');
  16. const request = require('../utils/request');
  17. const { Keyv } = require('keyv');
  18. const keyv = new Keyv();
  19. /**
  20. * 用户登录
  21. */
  22. router.post('/login', async (ctx) => {
  23. const { userName, userPwd, openId } = ctx.request.body;
  24. if (!userName || !userPwd) {
  25. util.fail(ctx, '用户名或密码不能为空');
  26. return;
  27. }
  28. const pwd = new md5().update(userPwd).digest('hex');
  29. const res = await userService.findUser(userName, pwd, openId || userName);
  30. if (!res) {
  31. util.fail(ctx, '用户名或密码错误');
  32. return;
  33. }
  34. if (!res.openId && openId) {
  35. const cacheUser = await keyv.get(openId);
  36. if (cacheUser) {
  37. await userService.bindOpenId({ ...cacheUser, id: res.id });
  38. }
  39. }
  40. const token = util.createToken({ userName, userId: res.id, nickName: res.nickName });
  41. userService.updateUserInfo(res.id);
  42. util.success(ctx, {
  43. userId: res.id,
  44. userName,
  45. token,
  46. });
  47. });
  48. /**
  49. * 微信授权登录
  50. */
  51. router.post('/wechat', async (ctx) => {
  52. const { code } = ctx.request.body;
  53. if (!code) {
  54. util.fail(ctx, 'code不能为空');
  55. return;
  56. }
  57. const response = await request.get(
  58. `https://api.weixin.qq.com/sns/oauth2/access_token?appid=${config.WECHAT_APP_ID}&secret=${config.WECHAT_APP_SECRET}&code=${code}&grant_type=authorization_code`,
  59. {},
  60. );
  61. const { access_token, openid, unionid, errcode } = JSON.parse(response.data);
  62. if (errcode) {
  63. util.success(ctx, '');
  64. return;
  65. }
  66. const wxUser = await userService.findUser(openid, openid, openid);
  67. if (wxUser) {
  68. userService.updateUserInfo(wxUser.id);
  69. const token = util.createToken({ userName: wxUser.userName, userId: wxUser.id, nickName: wxUser.nickName });
  70. util.success(ctx, {
  71. userId: wxUser.id,
  72. userName: wxUser.userName,
  73. token,
  74. });
  75. return;
  76. }
  77. const res1 = await request.get(`https://api.weixin.qq.com/sns/userinfo?access_token=${access_token}&openid=${openid}`);
  78. const { nickname, headimgurl } = JSON.parse(res1.data);
  79. await keyv.set(
  80. openid,
  81. {
  82. openid,
  83. unionid,
  84. nickname,
  85. headimgurl,
  86. },
  87. 3 * 60 * 1000,
  88. );
  89. util.success(ctx, {
  90. openId: openid,
  91. });
  92. });
  93. /**
  94. * 获取用户信息
  95. */
  96. router.get('/info', async (ctx) => {
  97. const { userId } = util.decodeToken(ctx);
  98. const res = await userService.profile(userId);
  99. util.success(ctx, res);
  100. });
  101. /**
  102. * 获取个人信息
  103. */
  104. router.get('/profile', async (ctx) => {
  105. const { userId } = util.decodeToken(ctx);
  106. const res = await userService.profile(userId);
  107. util.success(ctx, res);
  108. });
  109. /**
  110. * 用户搜索
  111. */
  112. router.post('/search', async (ctx) => {
  113. const { keyword } = ctx.request.body;
  114. if (!keyword) {
  115. util.fail(ctx, '关键字不能为空');
  116. return;
  117. }
  118. const res = await userService.search(keyword);
  119. if (!res) {
  120. util.fail(ctx, '当前用户名不存在');
  121. return;
  122. }
  123. util.success(ctx, res);
  124. });
  125. /**
  126. * 用户信息更新
  127. */
  128. router.post('/update/profile', async (ctx) => {
  129. const { nickName, avatar } = ctx.request.body;
  130. if (!nickName && !avatar) {
  131. util.fail(ctx, '参数异常,请重新提交');
  132. return;
  133. }
  134. const { userId } = util.decodeToken(ctx);
  135. await userService.updateUserInfo(userId, nickName, avatar);
  136. util.success(ctx, '更新成功');
  137. });
  138. /**
  139. * 用户注册 - 发送验证码
  140. */
  141. router.post('/sendEmail', async (ctx) => {
  142. try {
  143. const { email } = ctx.request.body;
  144. if (!email || !/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email)) {
  145. util.fail(ctx, '邮箱不能为空或格式错误');
  146. return;
  147. }
  148. const val = await keyv.get(email);
  149. if (val) {
  150. util.fail(ctx, '验证码已发送,请查收');
  151. return;
  152. }
  153. let transporter = nodemailer.createTransport({
  154. host: config.EMAIL_HOST,
  155. port: config.EMAIL_PORT,
  156. auth: {
  157. user: config.EMAIL_USER, // 你的Gmail地址
  158. pass: config.EMAIL_PASSWORD, // 你的Gmail密码或应用专用密码
  159. },
  160. });
  161. const random = Math.random().toString().slice(2, 7).replace(/^(0)+/, '1');
  162. let mailOptions = {
  163. from: `"Marsview" <${config.EMAIL_USER}>`, // 发送者地址
  164. to: email, // 接收者列表
  165. subject: 'Marsview账号注册', // 主题行
  166. text: '验证码发送', // 纯文字正文
  167. html: `当前验证码为:<b>${random}</b>,3分钟内有效。<br/><br/>感谢您体验 Marsview 搭建平台,线上平台不保证数据的稳定性,建议有条件用户,切换到 Marsview 私有化部署服务,您在使用过程中遇到任何问题均联系我。<br/><br/>邮 箱:marsview@163.com<br/>微 信:17611021717`, // HTML正文
  168. };
  169. await transporter.sendMail(mailOptions);
  170. await keyv.set(email, random, 3 * 60 * 1000);
  171. util.success(ctx, '发送成功');
  172. } catch (error) {
  173. util.fail(ctx, error.message);
  174. }
  175. });
  176. /**
  177. * 用户注册
  178. */
  179. router.post('/regist', async (ctx) => {
  180. const { userName, code, userPwd } = ctx.request.body;
  181. if (!userName || !userPwd) {
  182. util.fail(ctx, '用户名或密码不能为空');
  183. return;
  184. }
  185. if (!code) {
  186. util.fail(ctx, '邮箱验证码不能为空');
  187. return;
  188. }
  189. const val = await keyv.get(userName);
  190. if (!val) {
  191. util.fail(ctx, '验证码已过期');
  192. return;
  193. }
  194. if (val != code) {
  195. util.fail(ctx, '验证码错误');
  196. return;
  197. }
  198. const user = await userService.search(userName);
  199. if (user) {
  200. util.fail(ctx, '当前用户已存在');
  201. return;
  202. }
  203. const nickName = userName.split('@')[0];
  204. const pwd = new md5().update(userPwd).digest('hex');
  205. const res = await userService.create(nickName, userName, pwd);
  206. if (res.affectedRows == 1) {
  207. // 生成用户token
  208. const token = util.createToken({ userName, userId: res.insertId });
  209. util.success(ctx, {
  210. userId: res.id,
  211. userName,
  212. token,
  213. });
  214. } else {
  215. util.fail(ctx, '注册失败,请重试');
  216. }
  217. });
  218. /**
  219. * 忘记密码 - 生成链接
  220. */
  221. router.post('/password/forget', async (ctx) => {
  222. const { userEmail } = ctx.request.body;
  223. if (!userEmail) {
  224. util.fail(ctx, '邮箱不能为空');
  225. return;
  226. }
  227. const user = await userService.search(userEmail);
  228. if (!user) {
  229. util.fail(ctx, '当前用户不存在');
  230. return;
  231. }
  232. // 生成验证码,保存在redis中,用来验证链接有效期
  233. const random = Math.random().toString().slice(2, 7);
  234. await keyv.set(userEmail, random, 5 * 60 * 1000);
  235. // 生成加密后token
  236. const token = util.createToken({ userEmail });
  237. // 发送邮件
  238. let transporter = nodemailer.createTransport({
  239. host: config.EMAIL_HOST,
  240. port: config.EMAIL_PORT,
  241. auth: {
  242. user: config.EMAIL_USER, // 你的Gmail地址
  243. pass: config.EMAIL_PASSWORD, // 你的Gmail密码或应用专用密码
  244. },
  245. });
  246. let mailOptions = {
  247. from: `"Marsview" <${config.EMAIL_USER}>`, // 发送者地址
  248. to: userEmail, // 接收者列表
  249. subject: 'Marsview密码找回', // 主题行
  250. text: '验证码发送', // 纯文字正文
  251. html: `Hello,${userEmail}! <br/> 我们收到了你重置密码的申请,请点击下方按链接行重置,<a href="https://www.marsview.com.cn/password-reset?resetToken=${token}">重置密码</a> <br/> 链接 3分钟内有效,请尽快操作,如不是你发起的请求,请忽略。`, // HTML正文
  252. };
  253. await transporter.sendMail(mailOptions);
  254. util.success(ctx, '发送成功');
  255. });
  256. /**
  257. * 忘记密码 - 获取账号
  258. */
  259. router.post('/password/getUserByToken', async (ctx) => {
  260. const { resetToken } = ctx.request.query;
  261. const { userEmail } = util.decodeResetToken(resetToken);
  262. const val = await keyv.get(userEmail);
  263. if (!val) {
  264. util.fail(ctx, '链接已失效,请重新操作');
  265. return;
  266. }
  267. util.success(ctx, userEmail);
  268. });
  269. /**
  270. * 忘记密码 - 重置密码
  271. */
  272. router.post('/password/reset', async (ctx) => {
  273. const { resetToken, userPwd } = ctx.request.body;
  274. if (!resetToken) {
  275. util.fail(ctx, '重置Token不能为空');
  276. return;
  277. }
  278. if (!userPwd) {
  279. util.fail(ctx, '重置密码不能为空');
  280. return;
  281. }
  282. const { userEmail } = util.decodeResetToken(resetToken);
  283. if (!userEmail) {
  284. util.fail(ctx, 'Token 识别错误,请重新操作');
  285. return;
  286. }
  287. const val = await keyv.get(userEmail);
  288. if (!val) {
  289. util.fail(ctx, '链接已失效,请重新操作');
  290. return;
  291. }
  292. const pwd = new md5().update(userPwd).digest('hex');
  293. await userService.resetPwd(userEmail, pwd);
  294. await keyv.delete(userEmail);
  295. util.success(ctx, '更新成功');
  296. });
  297. /**
  298. * 密码修改
  299. */
  300. router.post('/password/update', async (ctx) => {
  301. const { oldPwd, userPwd, confirmPwd } = ctx.request.body;
  302. if (!oldPwd || !userPwd || !confirmPwd) {
  303. util.fail(ctx, '密码不能为空');
  304. return;
  305. }
  306. if (userPwd !== confirmPwd) {
  307. util.fail(ctx, '两次密码不一致');
  308. return;
  309. }
  310. const { userName } = util.decodeToken(ctx);
  311. try {
  312. const res = await userService.verifyOldPwd(userName, oldPwd);
  313. if (res) {
  314. const pwd = new md5().update(userPwd).digest('hex');
  315. await userService.resetPwd(userName, pwd);
  316. util.success(ctx, '密码更改成功');
  317. } else {
  318. util.fail(ctx, '原密码输入错误');
  319. }
  320. } catch (error) {
  321. util.fail(ctx, error.message);
  322. }
  323. });
  324. /**
  325. * 用户注销
  326. */
  327. router.post('/logout', async (ctx) => {
  328. const { userId, userName } = util.decodeToken(ctx);
  329. if (userId == 50) {
  330. util.fail(ctx, '管理员账号不支持注销');
  331. return;
  332. }
  333. // 删除用户所有项目
  334. await projectService.deleteAllProject(userId, userName);
  335. // 删除用户所有关联
  336. await projectUserService.deleteAllProjectUser(userId);
  337. // 删除用户所有页面角色
  338. await pageRoleService.deleteAllPageRole(userId);
  339. // 删除用户所有页面
  340. await pageService.deleteAllPage(userId, userName);
  341. // 删除用户所有菜单
  342. await menuService.deleteAllMenu(userId);
  343. // 删除用户所有发布
  344. await publishService.deleteAllPublish(userId);
  345. // 删除用户所有角色
  346. await roleService.deleteAllRole(userId);
  347. // 删除用户
  348. await userService.deleteUser(userId, userName);
  349. // 删除用户图片
  350. await imgcloudService.deleteAllImg(userId);
  351. util.success(ctx, '注销成功');
  352. });
  353. /**
  354. * 查询子用户列表
  355. */
  356. router.get('/subUser/list', async (ctx) => {
  357. const { userId } = util.decodeToken(ctx);
  358. const { pageNum, pageSize, keyword } = ctx.request.query;
  359. const { total } = await userService.getSubUsersCount(userId, keyword);
  360. if (total == 0) {
  361. return util.success(ctx, {
  362. list: [],
  363. total: 0,
  364. pageSize: +pageSize || 12,
  365. pageNum: +pageNum || 1,
  366. });
  367. }
  368. const list = await userService.getSubUsersList(userId, pageNum || 1, pageSize || 12, keyword);
  369. util.success(ctx, {
  370. list,
  371. total,
  372. pageSize: +pageSize,
  373. pageNum: +pageNum,
  374. });
  375. });
  376. // 创建子用户
  377. router.post('/subUser/create', async (ctx) => {
  378. const { userName, userPwd } = ctx.request.body;
  379. if (!userName || !userPwd) {
  380. util.fail(ctx, '用户名或密码不能为空');
  381. return;
  382. }
  383. const { userId } = util.decodeToken(ctx);
  384. const user = await userService.search(userName);
  385. if (user) {
  386. util.fail(ctx, '当前用户已存在');
  387. return;
  388. }
  389. const nickName = userName.split('@')[0];
  390. const pwd = new md5().update(userPwd).digest('hex');
  391. const res = await userService.create(nickName, userName, pwd, userId);
  392. if (res.affectedRows == 1) {
  393. util.success(ctx, '注册成功');
  394. } else {
  395. util.fail(ctx, '注册失败,请重试');
  396. }
  397. });
  398. // 删除子用户
  399. router.post('/subUser/delete', async (ctx) => {
  400. const { id } = ctx.request.body;
  401. if (!id) {
  402. util.fail(ctx, '用户ID不能为空');
  403. return;
  404. }
  405. const { userId } = util.decodeToken(ctx);
  406. const res = await userService.deleteSubUser(id, userId);
  407. if (res.affectedRows == 1) {
  408. util.success(ctx, '删除成功');
  409. } else {
  410. util.fail(ctx, '删除失败,请重试');
  411. }
  412. });
  413. module.exports = router;