me.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387
  1. module.exports = function(crowi, app) {
  2. 'use strict';
  3. var debug = require('debug')('growi:routes:me')
  4. , fs = require('fs')
  5. , models = crowi.models
  6. , config = crowi.getConfig()
  7. , User = models.User
  8. , UserGroupRelation = models.UserGroupRelation
  9. , ExternalAccount = models.ExternalAccount
  10. , ApiResponse = require('../util/apiResponse')
  11. //, pluginService = require('../service/plugin')
  12. , actions = {}
  13. , api = {}
  14. ;
  15. actions.api = api;
  16. /**
  17. * retrieve user-group-relation documents
  18. * @param {object} req
  19. * @param {object} res
  20. */
  21. api.userGroupRelations = function(req, res) {
  22. UserGroupRelation.findAllRelationForUser(req.user)
  23. .then(userGroupRelations => {
  24. return res.json(ApiResponse.success({userGroupRelations}));
  25. });
  26. };
  27. actions.index = function(req, res) {
  28. var userForm = req.body.userForm;
  29. var userData = req.user;
  30. if (req.method == 'POST' && req.form.isValid) {
  31. var name = userForm.name;
  32. var email = userForm.email;
  33. var lang = userForm.lang;
  34. var isEmailPublished = userForm.isEmailPublished;
  35. /*
  36. * disabled because the system no longer allows undefined email -- 2017.10.06 Yuki Takei
  37. *
  38. if (!User.isEmailValid(email)) {
  39. req.form.errors.push('You can\'t update to that email address');
  40. return res.render('me/index', {});
  41. }
  42. */
  43. User.findOneAndUpdate(
  44. { email: userData.email }, // query
  45. { name, email, lang, isEmailPublished }, // updating data
  46. { runValidators: true, context: 'query' }, // for validation
  47. // see https://www.npmjs.com/package/mongoose-unique-validator#find--updates -- 2017.09.24 Yuki Takei
  48. (err) => {
  49. if (err) {
  50. Object.keys(err.errors).forEach((e) => {
  51. req.form.errors.push(err.errors[e].message);
  52. });
  53. return res.render('me/index', {});
  54. }
  55. req.i18n.changeLanguage(lang);
  56. req.flash('successMessage', req.t('Updated'));
  57. return res.redirect('/me');
  58. });
  59. }
  60. else { // method GET
  61. /*
  62. * disabled because the system no longer allows undefined email -- 2017.10.06 Yuki Takei
  63. *
  64. /// そのうちこのコードはいらなくなるはず
  65. if (!userData.isEmailSet()) {
  66. req.flash('warningMessage', 'メールアドレスが設定されている必要があります');
  67. }
  68. */
  69. return res.render('me/index', {
  70. });
  71. }
  72. };
  73. actions.imagetype = function(req, res) {
  74. if (req.method !== 'POST') {
  75. // do nothing
  76. return;
  77. }
  78. else if (!req.form.isValid) {
  79. req.flash('errorMessage', req.form.errors.join('\n'));
  80. return;
  81. }
  82. var imagetypeForm = req.body.imagetypeForm;
  83. var userData = req.user;
  84. var isGravatarEnabled = imagetypeForm.isGravatarEnabled;
  85. userData.updateIsGravatarEnabled(isGravatarEnabled, function(err, userData) {
  86. if (err) {
  87. for (var e in err.errors) {
  88. if (err.errors.hasOwnProperty(e)) {
  89. req.form.errors.push(err.errors[e].message);
  90. }
  91. }
  92. return res.render('me/index', {});
  93. }
  94. req.flash('successMessage', req.t('Updated'));
  95. return res.redirect('/me');
  96. });
  97. };
  98. actions.externalAccounts = {};
  99. actions.externalAccounts.list = function(req, res) {
  100. const userData = req.user;
  101. let renderVars = {};
  102. ExternalAccount.find({user: userData})
  103. .then((externalAccounts) => {
  104. renderVars.externalAccounts = externalAccounts;
  105. return;
  106. })
  107. .then(() => {
  108. if (req.method == 'POST' && req.form.isValid) {
  109. // TODO impl
  110. return res.render('me/external-accounts', renderVars);
  111. }
  112. else { // method GET
  113. return res.render('me/external-accounts', renderVars);
  114. }
  115. });
  116. };
  117. actions.externalAccounts.disassociate = function(req, res) {
  118. const userData = req.user;
  119. const redirectWithFlash = (type, msg) => {
  120. req.flash(type, msg);
  121. return res.redirect('/me/external-accounts');
  122. };
  123. if (req.body == null) {
  124. redirectWithFlash('errorMessage', 'Invalid form.');
  125. }
  126. // make sure password set or this user has two or more ExternalAccounts
  127. new Promise((resolve, reject) => {
  128. if (userData.password != null) {
  129. resolve(true);
  130. }
  131. else {
  132. ExternalAccount.count({user: userData})
  133. .then((count) => {
  134. resolve(count > 1);
  135. });
  136. }
  137. })
  138. .then((isDisassociatable) => {
  139. if (!isDisassociatable) {
  140. let e = new Error();
  141. e.name = 'couldntDisassociateError';
  142. throw e;
  143. }
  144. const providerType = req.body.providerType;
  145. const accountId = req.body.accountId;
  146. return ExternalAccount.findOneAndRemove({providerType, accountId, user: userData});
  147. })
  148. .then((account) => {
  149. if (account == null) {
  150. return redirectWithFlash('errorMessage', 'ExternalAccount not found.');
  151. }
  152. else {
  153. return redirectWithFlash('successMessage', 'Successfully disassociated.');
  154. }
  155. })
  156. .catch((err) => {
  157. if (err) {
  158. if (err.name == 'couldntDisassociateError') {
  159. return redirectWithFlash('couldntDisassociateError', true);
  160. }
  161. else {
  162. return redirectWithFlash('errorMessage', err.message);
  163. }
  164. }
  165. });
  166. };
  167. actions.externalAccounts.associateLdap = function(req, res) {
  168. const passport = require('passport');
  169. const passportService = crowi.passportService;
  170. const redirectWithFlash = (type, msg) => {
  171. req.flash(type, msg);
  172. return res.redirect('/me/external-accounts');
  173. };
  174. if (!passportService.isLdapStrategySetup) {
  175. debug('LdapStrategy has not been set up');
  176. return redirectWithFlash('warning', 'LdapStrategy has not been set up');
  177. }
  178. const loginForm = req.body.loginForm;
  179. passport.authenticate('ldapauth', (err, user, info) => {
  180. if (res.headersSent) { // dirty hack -- 2017.09.25
  181. return; // cz: somehow passport.authenticate called twice when ECONNREFUSED error occurred
  182. }
  183. if (err) { // DB Error
  184. console.log('LDAP Server Error: ', err);
  185. return redirectWithFlash('warningMessage', 'LDAP Server Error occured.');
  186. }
  187. if (info && info.message) {
  188. return redirectWithFlash('warningMessage', info.message);
  189. }
  190. if (user) {
  191. // create ExternalAccount
  192. const ldapAccountId = passportService.getLdapAccountIdFromReq(req);
  193. const user = req.user;
  194. ExternalAccount.associate('ldap', ldapAccountId, user)
  195. .then(() => {
  196. return redirectWithFlash('successMessage', 'Successfully added.');
  197. })
  198. .catch((err) => {
  199. return redirectWithFlash('errorMessage', err.message);
  200. });
  201. }
  202. })(req, res, () => {});
  203. };
  204. actions.password = function(req, res) {
  205. var passwordForm = req.body.mePassword;
  206. var userData = req.user;
  207. /*
  208. * disabled because the system no longer allows undefined email -- 2017.10.06 Yuki Takei
  209. *
  210. // パスワードを設定する前に、emailが設定されている必要がある (schemaを途中で変更したため、最初の方の人は登録されていないかもしれないため)
  211. // そのうちこのコードはいらなくなるはず
  212. if (!userData.isEmailSet()) {
  213. return res.redirect('/me');
  214. }
  215. */
  216. if (req.method == 'POST' && req.form.isValid) {
  217. var newPassword = passwordForm.newPassword;
  218. var newPasswordConfirm = passwordForm.newPasswordConfirm;
  219. var oldPassword = passwordForm.oldPassword;
  220. if (userData.isPasswordSet() && !userData.isPasswordValid(oldPassword)) {
  221. req.form.errors.push('Wrong current password');
  222. return res.render('me/password', {
  223. });
  224. }
  225. // check password confirm
  226. if (newPassword != newPasswordConfirm) {
  227. req.form.errors.push('Failed to verify passwords');
  228. }
  229. else {
  230. userData.updatePassword(newPassword, function(err, userData) {
  231. if (err) {
  232. for (var e in err.errors) {
  233. if (err.errors.hasOwnProperty(e)) {
  234. req.form.errors.push(err.errors[e].message);
  235. }
  236. }
  237. return res.render('me/password', {});
  238. }
  239. req.flash('successMessage', 'Password updated');
  240. return res.redirect('/me/password');
  241. });
  242. }
  243. }
  244. else { // method GET
  245. return res.render('me/password', {
  246. });
  247. }
  248. };
  249. actions.apiToken = function(req, res) {
  250. var apiTokenForm = req.body.apiTokenForm;
  251. var userData = req.user;
  252. if (req.method == 'POST' && req.form.isValid) {
  253. userData.updateApiToken()
  254. .then(function(userData) {
  255. req.flash('successMessage', 'API Token updated');
  256. return res.redirect('/me/apiToken');
  257. })
  258. .catch(function(err) {
  259. //req.flash('successMessage',);
  260. req.form.errors.push('Failed to update API Token');
  261. return res.render('me/api_token', {
  262. });
  263. });
  264. }
  265. else {
  266. return res.render('me/api_token', {
  267. });
  268. }
  269. };
  270. actions.updates = function(req, res) {
  271. res.render('me/update', {
  272. });
  273. };
  274. actions.authGoogle = function(req, res) {
  275. var googleAuth = require('../util/googleAuth')(crowi);
  276. var userData = req.user;
  277. var toDisconnect = req.body.disconnectGoogle ? true : false;
  278. var toConnect = req.body.connectGoogle ? true : false;
  279. if (toDisconnect) {
  280. userData.deleteGoogleId(function(err, userData) {
  281. req.flash('successMessage', 'Disconnected from Google account');
  282. return res.redirect('/me');
  283. });
  284. }
  285. else if (toConnect) {
  286. googleAuth.createAuthUrl(req, function(err, redirectUrl) {
  287. if (err) {
  288. // TODO
  289. }
  290. req.session.googleCallbackAction = '/me/auth/google/callback';
  291. return res.redirect(redirectUrl);
  292. });
  293. }
  294. else {
  295. return res.redirect('/me');
  296. }
  297. };
  298. actions.authGoogleCallback = function(req, res) {
  299. var googleAuth = require('../util/googleAuth')(crowi);
  300. var userData = req.user;
  301. googleAuth.handleCallback(req, function(err, tokenInfo) {
  302. if (err) {
  303. req.flash('warningMessage.auth.google', err.message); // FIXME: show library error message directly
  304. return res.redirect('/me'); // TODO Handling
  305. }
  306. var googleId = tokenInfo.user_id;
  307. var googleEmail = tokenInfo.email;
  308. if (!User.isEmailValid(googleEmail)) {
  309. req.flash('warningMessage.auth.google', 'You can\'t connect with this Google\'s account');
  310. return res.redirect('/me');
  311. }
  312. User.findUserByGoogleId(googleId, function(err, googleUser) {
  313. if (!err && googleUser) {
  314. req.flash('warningMessage.auth.google', 'This Google\'s account is connected by another user');
  315. return res.redirect('/me');
  316. }
  317. else {
  318. userData.updateGoogleId(googleId, function(err, userData) {
  319. if (err) {
  320. debug('Failed to updateGoogleId', err);
  321. req.flash('warningMessage.auth.google', 'Failed to connect Google Account');
  322. return res.redirect('/me');
  323. }
  324. // TODO if err
  325. req.flash('successMessage', 'Connected with Google');
  326. return res.redirect('/me');
  327. });
  328. }
  329. });
  330. });
  331. };
  332. return actions;
  333. };