activity.ts 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. import { DeleteWriteOpResultObject } from 'mongodb';
  2. import {
  3. Types, Document, Model, Schema,
  4. } from 'mongoose';
  5. import { getOrCreateModel, getModelSafely } from '../util/mongoose-utils';
  6. import loggerFactory from '../../utils/logger';
  7. import ActivityDefine from '../util/activityDefine';
  8. import Watcher from './watcher';
  9. // import { InAppNotification } from './in-app-notification';
  10. // import activityEvent from '../events/activity';
  11. const logger = loggerFactory('growi:models:activity');
  12. export interface ActivityDocument extends Document {
  13. _id: Types.ObjectId
  14. user: Types.ObjectId | any
  15. targetModel: string
  16. target: string
  17. action: string
  18. event: Types.ObjectId
  19. eventModel: string
  20. createdAt: Date
  21. getNotificationTargetUsers(): Promise<any[]>
  22. }
  23. export interface ActivityModel extends Model<ActivityDocument> {
  24. createByParameters(parameters: any): Promise<ActivityDocument>
  25. removeByParameters(parameters: any): any
  26. createByPageComment(comment: any): Promise<ActivityDocument>
  27. createByPageLike(page: any, user: any): Promise<ActivityDocument>
  28. removeByPageUnlike(page: any, user: any): Promise<DeleteWriteOpResultObject['result']>
  29. removeByPage(page: any): Promise<DeleteWriteOpResultObject['result']>
  30. findByUser(user: any): Promise<ActivityDocument[]>
  31. getActionUsersFromActivities(activities: ActivityDocument[]): any[]
  32. }
  33. // const activityEvent = crowi.event('Activity');
  34. // TODO: add revision id
  35. const activitySchema = new Schema<ActivityDocument, ActivityModel>({
  36. user: {
  37. type: Schema.Types.ObjectId,
  38. ref: 'User',
  39. index: true,
  40. require: true,
  41. },
  42. targetModel: {
  43. type: String,
  44. require: true,
  45. enum: ActivityDefine.getSupportTargetModelNames(),
  46. },
  47. target: {
  48. type: Schema.Types.ObjectId,
  49. refPath: 'targetModel',
  50. require: true,
  51. },
  52. action: {
  53. type: String,
  54. require: true,
  55. enum: ActivityDefine.getSupportActionNames(),
  56. },
  57. event: {
  58. type: Schema.Types.ObjectId,
  59. refPath: 'eventModel',
  60. },
  61. eventModel: {
  62. type: String,
  63. enum: ActivityDefine.getSupportEventModelNames(),
  64. },
  65. createdAt: {
  66. type: Date,
  67. default: Date.now,
  68. },
  69. });
  70. activitySchema.index({ target: 1, action: 1 });
  71. activitySchema.index({
  72. user: 1, target: 1, action: 1, createdAt: 1,
  73. }, { unique: true });
  74. /**
  75. * @param {object} parameters
  76. * @return {Promise}
  77. */
  78. activitySchema.statics.createByParameters = function(parameters) {
  79. return this.create(parameters);
  80. };
  81. /**
  82. * @param {object} parameters
  83. */
  84. activitySchema.statics.removeByParameters = async function(parameters) {
  85. const activity = await this.findOne(parameters);
  86. // activityEvent.emit('remove', activity);
  87. return this.deleteMany(parameters).exec();
  88. };
  89. /**
  90. * @param {Comment} comment
  91. * @return {Promise}
  92. */
  93. activitySchema.statics.createByPageComment = function(comment) {
  94. const parameters = {
  95. user: comment.creator,
  96. targetModel: ActivityDefine.MODEL_PAGE,
  97. target: comment.page,
  98. eventModel: ActivityDefine.MODEL_COMMENT,
  99. event: comment._id,
  100. action: ActivityDefine.ACTION_COMMENT,
  101. };
  102. return this.createByParameters(parameters);
  103. };
  104. /**
  105. * @param {Page} page
  106. * @param {User} user
  107. * @return {Promise}
  108. */
  109. activitySchema.statics.createByPageLike = function(page, user) {
  110. const parameters = {
  111. user: user._id,
  112. targetModel: ActivityDefine.MODEL_PAGE,
  113. target: page,
  114. action: ActivityDefine.ACTION_LIKE,
  115. };
  116. return this.createByParameters(parameters);
  117. };
  118. /**
  119. * @param {Page} page
  120. * @param {User} user
  121. * @return {Promise}
  122. */
  123. activitySchema.statics.removeByPageUnlike = function(page, user) {
  124. const parameters = {
  125. user,
  126. targetModel: ActivityDefine.MODEL_PAGE,
  127. target: page,
  128. action: ActivityDefine.ACTION_LIKE,
  129. };
  130. return this.removeByParameters(parameters);
  131. };
  132. /**
  133. * @param {Page} page
  134. * @return {Promise}
  135. */
  136. activitySchema.statics.removeByPage = async function(page) {
  137. const activities = await this.find({ target: page });
  138. for (const activity of activities) {
  139. // activityEvent.emit('remove', activity);
  140. }
  141. return this.deleteMany({ target: page }).exec();
  142. };
  143. /**
  144. * @param {User} user
  145. * @return {Promise}
  146. */
  147. activitySchema.statics.findByUser = function(user) {
  148. return this.find({ user }).sort({ createdAt: -1 }).exec();
  149. };
  150. activitySchema.statics.getActionUsersFromActivities = function(activities) {
  151. return activities.map(({ user }) => user).filter((user, i, self) => self.indexOf(user) === i);
  152. };
  153. activitySchema.methods.getNotificationTargetUsers = async function() {
  154. const User = getModelSafely('User') || require('~/server/models/user')();
  155. const { user: actionUser, targetModel, target } = this;
  156. const model: any = await this.model(targetModel).findById(target);
  157. const [targetUsers, watchUsers, ignoreUsers] = await Promise.all([
  158. model.getNotificationTargetUsers(),
  159. Watcher.getWatchers((target as any) as Types.ObjectId),
  160. Watcher.getIgnorers((target as any) as Types.ObjectId),
  161. ]);
  162. const unique = array => Object.values(array.reduce((objects, object) => ({ ...objects, [object.toString()]: object }), {}));
  163. const filter = (array, pull) => {
  164. const ids = pull.map(object => object.toString());
  165. return array.filter(object => !ids.includes(object.toString()));
  166. };
  167. const notificationUsers = filter(unique([...targetUsers, ...watchUsers]), [...ignoreUsers, actionUser]);
  168. const activeNotificationUsers = await User.find({
  169. _id: { $in: notificationUsers },
  170. status: User.STATUS_ACTIVE,
  171. }).distinct('_id');
  172. return activeNotificationUsers;
  173. };
  174. /**
  175. * saved hook
  176. */
  177. activitySchema.post('save', async(savedActivity: ActivityDocument) => {
  178. try {
  179. const notificationUsers = await savedActivity.getNotificationTargetUsers();
  180. // await Promise.all(notificationUsers.map(user => InAppNotification.upsertByActivity(user, savedActivity)));
  181. return;
  182. }
  183. catch (err) {
  184. logger.error(err);
  185. }
  186. });
  187. /**
  188. * TODO: improve removeActivity that decleard in InAppNotificationService by GW-7481
  189. */
  190. // because mongoose's 'remove' hook fired only when remove by a method of Document (not by a Model method)
  191. // move 'save' hook from mongoose's events to activityEvent if I have a time.
  192. // activityEvent.on('remove', async(activity: ActivityDocument) => {
  193. // try {
  194. // await InAppNotification.removeActivity(activity);
  195. // }
  196. // catch (err) {
  197. // logger.error(err);
  198. // }
  199. // });
  200. const Activity = getOrCreateModel<ActivityDocument, ActivityModel>('Activity', activitySchema);
  201. export { Activity };