From 14c75eefd0418a6197abb0725b190cb6eb60c974 Mon Sep 17 00:00:00 2001 From: Huynh Manh Tuong Date: Sun, 28 Jan 2024 09:14:59 +0700 Subject: [PATCH] update API Like and Bookmark --- .env.example | 1 + openapi/components.yaml | 67 ++- openapi/paths.yaml | 429 +++++++++++++++++++ package.json | 2 +- src/constants/config.ts | 1 + src/constants/messages.ts | 23 +- src/controllers/bookmarks.controller.ts | 18 + src/controllers/likes.controller.ts | 19 + src/controllers/tweets.controllers.ts | 7 +- src/controllers/users.controllers.ts | 30 ++ src/models/requests/Notification.requests.ts | 9 + src/models/requests/Tweet.requests.ts | 3 + src/models/requests/User.requests.ts | 6 +- src/models/schemas/Notifications.schema.ts | 38 ++ src/routes/bookmarks.routes.ts | 18 +- src/routes/likes.routes.ts | 17 +- src/routes/users.routes.ts | 33 +- src/services/bookmarks.services.ts | 287 +++++++++++++ src/services/database.services.ts | 4 + src/services/likes.services.ts | 287 +++++++++++++ src/services/notifications.services.ts | 22 + src/services/tweets.services.ts | 23 +- src/services/users.services.ts | 191 +++++++++ src/utils/socket.ts | 1 - 24 files changed, 1516 insertions(+), 20 deletions(-) create mode 100644 src/models/requests/Notification.requests.ts create mode 100644 src/models/schemas/Notifications.schema.ts create mode 100644 src/services/notifications.services.ts diff --git a/.env.example b/.env.example index 90341f9..8d3e461 100644 --- a/.env.example +++ b/.env.example @@ -12,6 +12,7 @@ DB_HASHTAGS_COLLECTION="hashtags" DB_BOOKMARKS_COLLECTION="bookmarks" DB_LIKES_COLLECTION="likes" DB_CONVERSATIONS_COLLECTION="conversations" +DB_NOTIFICATIONS_COLLECTION="notifications" PASSWORD_SECRET= '' JWT_SECRET_ACCESS_TOKEN = '' diff --git a/openapi/components.yaml b/openapi/components.yaml index 3308502..746f7ec 100644 --- a/openapi/components.yaml +++ b/openapi/components.yaml @@ -5,10 +5,10 @@ components: properties: email: type: string - example: admin1@gmail.com + example: hmt1@gmail.com password: type: string - example: Duoc123! + example: '@Aa1234567' SuccessAuthentication: type: object properties: @@ -298,3 +298,66 @@ components: type: string format: date-time description: Thời gian cập nhật tweet + LikeRequestBody: + type: object + properties: + tweet_id: + type: string + format: MongoId + description: Tweet ID + example: 64be0ad2e43d2464394feedb + BookmarkRequestBody: + type: object + properties: + tweet_id: + type: string + format: MongoId + description: Tweet ID + example: 64be0ad2e43d2464394feedb + UnTweetResponseBody: + type: object + properties: + _id: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + tweet_id: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + user_id: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + created_at: + type: string + example: 2021-09-30T15:00:00.000Z + Conversation: + type: object + properties: + _id: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + sender_id: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + receiver_id: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + content: + type: string + example: 'Hello world!' + created_at: + type: string + example: 2021-09-30T15:00:00.000Z + updated_at: + type: string + example: 2021-09-30T15:00:00.000Z + securitySchemes: + BearerAuth: + type: http + scheme: bearer + bearerFormat: JWT diff --git a/openapi/paths.yaml b/openapi/paths.yaml index 0494292..1955d09 100644 --- a/openapi/paths.yaml +++ b/openapi/paths.yaml @@ -669,3 +669,432 @@ paths: example: Upload video thành công result: $ref: '#/components/schemas/Media' + /search: + get: + tags: + - search + summary: List Of Tweets + description: Search Tweets by content, media type + operationId: searchtweets + security: + - BearerAuth: [] + parameters: + - name: content + in: query + description: Content Of Tweet + required: false + schema: + type: string + - name: page + in: query + description: Số trang + required: false + schema: + type: integer + example: 1 + - name: limit + in: query + description: Số lượng tweet trên 1 trang + required: false + schema: + type: integer + example: 10 + - name: media_type + in: query + description: Type Of Tweet + required: false + schema: + type: string + example: image + enum: + - image + - video + - name: people_follow + in: query + description: Tweet of my followers + required: false + schema: + type: boolean + example: true + responses: + '200': + description: search successful + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: search successful + result: + type: object + properties: + tweets: + type: array + items: + $ref: '#/components/schemas/Tweet' + total_page: + type: integer + example: 1 + limit: + type: integer + example: 10 + page: + type: integer + example: 1 + '400': + description: Invalid status value + /likes: + post: + tags: + - likes + summary: Like tweet + description: Like tweet + operationId: likeTweet + security: + - BearerAuth: [] + requestBody: + description: Tweet ID + content: + application/json: + schema: + $ref: '#/components/schemas/LikeRequestBody' + required: true + responses: + default: + description: Like tweet successful + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: LIKE Tweet successfully + result: + $ref: '#/components/schemas/UnTweetResponseBody' + get: + tags: + - likes + summary: Get Tweets User Liked + description: Get Tweets User Liked + operationId: tweetUserLiked + security: + - BearerAuth: [] + parameters: + - name: page + in: query + description: Số trang + required: false + schema: + type: integer + example: 1 + - name: limit + in: query + description: Số lượng tweet trên 1 trang + required: false + schema: + type: integer + example: 10 + responses: + '200': + description: Get tweets liked successfully + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Get tweets liked successfully + result: + type: object + properties: + tweets: + type: array + items: + $ref: '#/components/schemas/Tweet' + total_page: + type: integer + example: 1 + limit: + type: integer + example: 10 + page: + type: integer + example: 1 + '400': + description: Invalid status value + /likes/tweets/{tweet_id}: + delete: + tags: + - likes + summary: Unlike tweet by tweet_id + description: Unlike tweet by tweet_id + operationId: unlikeTweet + security: + - BearerAuth: [] + parameters: + - name: tweet_id + in: path + description: Tweet Id + required: true + schema: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + responses: + default: + description: Unlike successfully + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Unlike successfully + result: + $ref: '#/components/schemas/UnTweetResponseBody' + /likes/{like_id}: + delete: + tags: + - likes + summary: Unlike tweet by like_id + description: Unlike tweet by like_id + operationId: unlikeTweet + security: + - BearerAuth: [] + parameters: + - name: like_id + in: path + description: Like Id + required: true + schema: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + responses: + default: + description: Unlike successfully + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Unlike successfully + result: + $ref: '#/components/schemas/UnTweetResponseBody' + + /bookmarks: + post: + tags: + - bookmarks + summary: Bookmark tweet + description: Bookmark tweet + operationId: bookmarkTweet + security: + - BearerAuth: [] + requestBody: + description: Tweet ID + content: + application/json: + schema: + $ref: '#/components/schemas/BookmarkRequestBody' + required: true + responses: + default: + description: Bookmark tweet successful + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Bookmark Tweet successfully + result: + $ref: '#/components/schemas/UnTweetResponseBody' + get: + tags: + - bookmarks + summary: Get Tweets User Bookmarked + description: Get Tweets User Bookmarked + operationId: tweetUserBookmarked + security: + - BearerAuth: [] + parameters: + - name: page + in: query + description: Số trang + required: false + schema: + type: integer + example: 1 + - name: limit + in: query + description: Số lượng tweet trên 1 trang + required: false + schema: + type: integer + example: 10 + responses: + '200': + description: Get tweets bookmarked successfully + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Get tweets bookmarked successfully + result: + type: object + properties: + tweets: + type: array + items: + $ref: '#/components/schemas/Tweet' + total_page: + type: integer + example: 1 + limit: + type: integer + example: 10 + page: + type: integer + example: 1 + '400': + description: Invalid status value + /bookmarks/tweets/{tweet_id}: + delete: + tags: + - bookmarks + summary: Unbookmark tweet by tweet_id + description: Unbookmark tweet by tweet_id + operationId: unbookmarkTweet + security: + - BearerAuth: [] + parameters: + - name: tweet_id + in: path + description: Tweet Id + required: true + schema: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + responses: + default: + description: Unbookmark successfully + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Unbookmark successfully + result: + $ref: '#/components/schemas/UnTweetResponseBody' + /bookmarks/{bookmark_id}: + delete: + tags: + - bookmarks + summary: Unbookmark tweet by like_id + description: Unbookmark tweet by like_id + operationId: unbookmarkTweet + security: + - BearerAuth: [] + parameters: + - name: bookmark_id + in: path + description: Bookmark Id + required: true + schema: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + responses: + default: + description: Unbookmark successfully + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Unlike successfully + result: + $ref: '#/components/schemas/UnTweetResponseBody' + + /conversations/receivers/{receiver_id}: + get: + tags: + - conversations + summary: Get Conversations By receiverID + description: Get Conversations By receiverID + operationId: getConversationsByReceiverID + security: + - BearerAuth: [] + parameters: + - name: receiver_id + in: path + description: Receiver Id + required: true + schema: + type: string + format: MongoId + example: 64be0ad2e43d2464394feedb + - name: page + in: query + description: Số trang + required: false + schema: + type: integer + example: 1 + - name: limit + in: query + description: Số lượng tweet trên 1 trang + required: false + schema: + type: integer + example: 10 + responses: + '200': + description: Get conversations successfully + content: + application/json: + schema: + type: object + properties: + message: + type: string + example: Get conversations successfully + conversations: + type: array + items: + $ref: '#/components/schemas/Conversation' + result: + type: object + properties: + page: + type: integer + example: 1 + total_pages: + type: integer + example: 1 + limit: + type: integer + example: 10 + total_items: + type: integer + example: 1 + + '400': + description: Invalid status value diff --git a/package.json b/package.json index e012e04..6be74d5 100644 --- a/package.json +++ b/package.json @@ -4,7 +4,7 @@ "description": "", "main": "index.js", "scripts": { - "dev": "NODE_ENV=development npx nodemon", + "dev": "SET NODE_ENV=development & npx nodemon", "dev:stag": "NODE_ENV=staging npx nodemon", "dev:prod": "NODE_ENV=production npx nodemon", "build": "rimraf ./dist && tsc && tsc-alias", diff --git a/src/constants/config.ts b/src/constants/config.ts index 0d9e745..560ddde 100644 --- a/src/constants/config.ts +++ b/src/constants/config.ts @@ -47,6 +47,7 @@ export const envConfig = { dbFollowersCollection: process.env.DB_FOLLOWERS_COLLECTION as string, dbVideoStatusCollection: process.env.DB_VIDEO_STATUS_COLLECTION as string, dbConversationCollection: process.env.DB_CONVERSATIONS_COLLECTION as string, + dbNotificationCollection: process.env.DB_NOTIFICATIONS_COLLECTION as string, passwordSecret: process.env.PASSWORD_SECRET as string, jwtSecretAccessToken: process.env.JWT_SECRET_ACCESS_TOKEN as string, jwtSecretRefreshToken: process.env.JWT_SECRET_REFRESH_TOKEN as string, diff --git a/src/constants/messages.ts b/src/constants/messages.ts index 1fd9af9..d679595 100644 --- a/src/constants/messages.ts +++ b/src/constants/messages.ts @@ -64,7 +64,9 @@ export const USERS_MESSAGES = { CHANGE_PASSWORDS_SUCCESS: 'Change passwords successful', GMAIL_NOT_VERIFIED: 'Email not verified', UPLOAD_SUCCESS: 'Upload successful', - GET_VIDEO_STATUS_SUCCESS: 'Get video status successful' + GET_VIDEO_STATUS_SUCCESS: 'Get video status successful', + GET_RANDOM_USERS_SUCCESS: 'Get random user successful', + GET_FOLLOWER_SUCCESS: 'Get followers successful' } as const export const TWEETS_MESSAGES = { @@ -93,7 +95,8 @@ export const BOOKMARKS_MESSAGES = { TWEET_ID_NOT_EMPTY: 'The tweet ID must be not empty', INVALID_TWEET_ID: 'The tweet ID is invalid', TWEET_NOT_FOUND: 'The tweet is not found', - UNBOOKMARKS_SUCCESSFULLY: 'Unbookmark successfully' + UNBOOKMARKS_SUCCESSFULLY: 'Unbookmark successfully', + GET_BOOKMARKS_SUCCESSFULLY: 'Get books successfully' } as const export const LIKES_MESSAGES = { LIKES_SUCCESSFULLY: 'LIKE Tweet successfully', @@ -101,7 +104,8 @@ export const LIKES_MESSAGES = { TWEET_ID_NOT_EMPTY: 'The tweet ID must be not empty', INVALID_TWEET_ID: 'The tweet ID is invalid', TWEET_NOT_FOUND: 'The tweet is not found', - UNLIKE_SUCCESSFULLY: 'Unlike successfully' + UNLIKE_SUCCESSFULLY: 'Unlike successfully', + GET_TWEETS_LIKED_SUCCESSFULLY: 'Get tweets liked successfully' } as const export const SEARCH_MESSAGES = { SEARCH_SUCCESSFULLY: 'Search successfully', @@ -112,3 +116,16 @@ export const SEARCH_MESSAGES = { export const CONVERSATIONS_MESSAGES = { GET_CONVERSATIONS_SUCCESSFULLY: 'Get conversions successfully' } as const + +export const NOTIFICATION_MESSAGES = { + LIKE: (interacted_user_name: string, tweet_content: string) => + `${interacted_user_name} liked your tweet: ${tweet_content}`, + BOOKMARK: (interacted_user_name: string, tweet_content: string) => + `${interacted_user_name} bookmarked your tweet: ${tweet_content}`, + COMMENT: (interacted_user_name: string, tweet_content: string) => + `${interacted_user_name} commented your tweet: ${tweet_content}`, + RETWEET: (interacted_user_name: string, tweet_content: string) => + `${interacted_user_name} retweeted your tweet: ${tweet_content}`, + QUOTETWEET: (interacted_user_name: string, tweet_content: string) => + `${interacted_user_name} quoted your tweet: ${tweet_content}` +} as const diff --git a/src/controllers/bookmarks.controller.ts b/src/controllers/bookmarks.controller.ts index 6f935d4..790f3d8 100644 --- a/src/controllers/bookmarks.controller.ts +++ b/src/controllers/bookmarks.controller.ts @@ -2,9 +2,27 @@ import { NextFunction, Request, Response } from 'express' import { ParamsDictionary } from 'express-serve-static-core' import { BOOKMARKS_MESSAGES } from '~/constants/messages' import { BookmarkTweetRequestBody } from '~/models/requests/Bookmark.requests' +import { Pagination } from '~/models/requests/Tweet.requests' import { TokenPayload } from '~/models/requests/User.requests' import bookmarkService from '~/services/bookmarks.services' +export const getBookmarkController = async (req: Request, res: Response) => { + const { user_id } = req.decoded_authorization as TokenPayload + const limit = Number(req.query.limit) + const page = Number(req.query.page) + const { tweets, total } = await bookmarkService.getBookmarks({ user_id, limit, page }) + return res.json({ + message: BOOKMARKS_MESSAGES.GET_BOOKMARKS_SUCCESSFULLY, + result: { + tweets: tweets, + page: page, + limit: limit, + total_items: total, + total_pages: Math.ceil(total / limit) + } + }) +} + export const bookmarkTweetController = async ( req: Request, res: Response diff --git a/src/controllers/likes.controller.ts b/src/controllers/likes.controller.ts index a10e6ce..2429b70 100644 --- a/src/controllers/likes.controller.ts +++ b/src/controllers/likes.controller.ts @@ -2,8 +2,27 @@ import { NextFunction, Request, Response } from 'express' import { ParamsDictionary } from 'express-serve-static-core' import { LIKES_MESSAGES } from '~/constants/messages' import { LikeTweetRequestBody } from '~/models/requests/Like.requests' +import { Pagination } from '~/models/requests/Tweet.requests' import { TokenPayload } from '~/models/requests/User.requests' import likeService from '~/services/likes.services' + +export const getTweetUserLikedController = async (req: Request, res: Response) => { + const { user_id } = req.decoded_authorization as TokenPayload + const limit = Number(req.query.limit) + const page = Number(req.query.page) + const { tweets, total } = await likeService.getTweetUserLiked({ user_id, limit, page }) + return res.json({ + message: LIKES_MESSAGES.GET_TWEETS_LIKED_SUCCESSFULLY, + result: { + tweets: tweets, + page: page, + limit: limit, + total_items: total, + total_pages: Math.ceil(total / limit) + } + }) +} + export const likeTweetController = async (req: Request, res: Response) => { const { tweet_id } = req.body const { user_id } = req.decoded_authorization as TokenPayload diff --git a/src/controllers/tweets.controllers.ts b/src/controllers/tweets.controllers.ts index 7b30fca..f677f97 100644 --- a/src/controllers/tweets.controllers.ts +++ b/src/controllers/tweets.controllers.ts @@ -2,7 +2,7 @@ import { NextFunction, Request, Response } from 'express' import { ParamsDictionary } from 'express-serve-static-core' import { TweetType } from '~/constants/enums' import { TWEETS_MESSAGES } from '~/constants/messages' -import { Pagination, TweetParam, TweetQuery, TweetRequestBody } from '~/models/requests/Tweet.requests' +import { NewFeedQuery, Pagination, TweetParam, TweetQuery, TweetRequestBody } from '~/models/requests/Tweet.requests' import { TokenPayload } from '~/models/requests/User.requests' import tweetsService from '~/services/tweets.services' @@ -54,11 +54,12 @@ export const getTweetChildrenController = async (req: Request, res: Response) => { +export const getNewFeedsController = async (req: Request, res: Response) => { const user_id = req.decoded_authorization?.user_id as string + const isForYou = Boolean(req.query.isForYou) const limit = Number(req.query.limit) const page = Number(req.query.page) - const result = await tweetsService.getNewFeeds({ user_id, limit, page }) + const result = await tweetsService.getNewFeeds({ user_id, limit, page, isForYou }) return res.json({ message: TWEETS_MESSAGES.GET_NEW_FEED_SUCCESSFULLY, result: { diff --git a/src/controllers/users.controllers.ts b/src/controllers/users.controllers.ts index 2d4459c..1214b2b 100644 --- a/src/controllers/users.controllers.ts +++ b/src/controllers/users.controllers.ts @@ -25,6 +25,7 @@ import HTTP_STATUS from '~/constants/httpStatus' import { UserVerifyStatus } from '~/constants/enums' import { omit, pick } from 'lodash' import { envConfig } from '~/constants/config' +import { Pagination } from '~/models/requests/Tweet.requests' export const loginController = async (req: Request, res: Response) => { const user = req.user as User const user_id = user._id as ObjectId @@ -186,6 +187,7 @@ export const getProfileController = async (req: Request, re result: user }) } + export const followController = async ( req: Request, res: Response, @@ -202,6 +204,26 @@ export const unFollowController = async (req: Request, res: R const result = await usersService.unFollow(user_id, followed_user_id) return res.json(result) } +export const getFollowerController = async ( + req: Request, + res: Response, + next: NextFunction +) => { + const { user_id } = req.decoded_authorization as TokenPayload + const limit = Number(req.query.limit) + const page = Number(req.query.page) + const { followers, total } = await usersService.getFollower({ user_id, limit, page }) + return res.json({ + message: USERS_MESSAGES.GET_FOLLOWER_SUCCESS, + result: { + followers: followers, + page: page, + limit: limit, + total_items: total, + total_pages: Math.ceil(total / limit) + } + }) +} export const changePasswordController = async ( req: Request, @@ -213,3 +235,11 @@ export const changePasswordController = async ( const result = await usersService.changePassword(user_id, new_password) return res.json(result) } +export const getRandomUserController = async (req: Request, res: Response, next: NextFunction) => { + const users = await usersService.getRandomUser() + return res.json({ + message: USERS_MESSAGES.GET_RANDOM_USERS_SUCCESS, + total_users: 10, + result: users + }) +} diff --git a/src/models/requests/Notification.requests.ts b/src/models/requests/Notification.requests.ts new file mode 100644 index 0000000..935397b --- /dev/null +++ b/src/models/requests/Notification.requests.ts @@ -0,0 +1,9 @@ +import { TweetType } from '~/constants/enums' + +export interface NotificationRequestBody { + message: string + owner_username: string + interacted_username: string + tweet_id: string + children_tweet: string +} diff --git a/src/models/requests/Tweet.requests.ts b/src/models/requests/Tweet.requests.ts index 03b6cdc..630678f 100644 --- a/src/models/requests/Tweet.requests.ts +++ b/src/models/requests/Tweet.requests.ts @@ -33,3 +33,6 @@ export interface NewFeed extends Tweet { comment_count: number quote_count: number } +export interface NewFeedQuery extends Pagination, Query { + isForYou: string +} diff --git a/src/models/requests/User.requests.ts b/src/models/requests/User.requests.ts index 2358c11..9576b6a 100644 --- a/src/models/requests/User.requests.ts +++ b/src/models/requests/User.requests.ts @@ -1,7 +1,7 @@ import { JwtPayload } from 'jsonwebtoken' import { TokenType, UserVerifyStatus } from '~/constants/enums' -import { ParamsDictionary } from 'express-serve-static-core' - +import { ParamsDictionary, Query } from 'express-serve-static-core' +import { Pagination } from './Tweet.requests' /** * @swagger * components: @@ -147,3 +147,5 @@ export interface ChangePasswordReqBody { new_password: string confirm_password: string } + +export interface FollowerQuery extends Pagination, Query {} diff --git a/src/models/schemas/Notifications.schema.ts b/src/models/schemas/Notifications.schema.ts new file mode 100644 index 0000000..ec21e05 --- /dev/null +++ b/src/models/schemas/Notifications.schema.ts @@ -0,0 +1,38 @@ +import { ObjectId } from 'mongodb' + +interface NotificationType { + _id?: ObjectId + message: string + owner_username: string + interacted_username: string + tweet_id: ObjectId + children_tweet?: ObjectId | null + created_at?: Date +} + +export default class Notification { + _id?: ObjectId + message: string + owner_username: string + interacted_username: string + tweet_id: ObjectId + children_tweet?: ObjectId | null + created_at?: Date + constructor({ + _id, + message, + owner_username, + interacted_username, + tweet_id, + children_tweet, + created_at + }: NotificationType) { + this._id = _id || new ObjectId() + this.message = message + this.owner_username = owner_username + this.interacted_username = interacted_username + this.tweet_id = tweet_id + this.children_tweet = children_tweet + this.created_at = created_at || new Date() + } +} diff --git a/src/routes/bookmarks.routes.ts b/src/routes/bookmarks.routes.ts index 911d2d4..f1e0b9f 100644 --- a/src/routes/bookmarks.routes.ts +++ b/src/routes/bookmarks.routes.ts @@ -1,15 +1,31 @@ import { Router } from 'express' import { bookmarkTweetController, + getBookmarkController, unBookmarkTweetByBookmarkIdController, unBookmarkTweetController } from '~/controllers/bookmarks.controller' -import { tweetIdValidator } from '~/middlewares/tweets.middlewares' +import { paginationNavigator, tweetIdValidator } from '~/middlewares/tweets.middlewares' import { accessTokenValidator, verifiedUSerValidator } from '~/middlewares/users.middlewares' import { wrapRequestHandler } from '~/utils/handles' const bookmarksRouter = Router() +/** + * Description: Get My Bookmark + * Path: / + * Method: GET + * Header:{Authorization:Bearer } + * Body: + * **/ +bookmarksRouter.get( + '', + accessTokenValidator, + verifiedUSerValidator, + paginationNavigator, + wrapRequestHandler(getBookmarkController) +) + /** * Description: Create Bookmark * Path: / diff --git a/src/routes/likes.routes.ts b/src/routes/likes.routes.ts index 894a761..4a3c91e 100644 --- a/src/routes/likes.routes.ts +++ b/src/routes/likes.routes.ts @@ -5,16 +5,31 @@ import { unBookmarkTweetController } from '~/controllers/bookmarks.controller' import { + getTweetUserLikedController, likeTweetController, unlikeTweetByLikeIdController, unlikeTweetController } from '~/controllers/likes.controller' -import { tweetIdValidator } from '~/middlewares/tweets.middlewares' +import { paginationNavigator, tweetIdValidator } from '~/middlewares/tweets.middlewares' import { accessTokenValidator, verifiedUSerValidator } from '~/middlewares/users.middlewares' import { wrapRequestHandler } from '~/utils/handles' const likesRouter = Router() +/** + * Description: Get Tweet User Liked + * Path: / + * Method: GET + * Header:{Authorization:Bearer } + * Body: + * **/ +likesRouter.get( + '', + accessTokenValidator, + verifiedUSerValidator, + paginationNavigator, + wrapRequestHandler(getTweetUserLikedController) +) /** * Description: Create Like * Path: / diff --git a/src/routes/users.routes.ts b/src/routes/users.routes.ts index 9b7dfdb..452f147 100644 --- a/src/routes/users.routes.ts +++ b/src/routes/users.routes.ts @@ -3,8 +3,10 @@ import { changePasswordController, followController, forgotPasswordController, + getFollowerController, getMeController, getProfileController, + getRandomUserController, loginController, logoutController, oauthController, @@ -18,6 +20,7 @@ import { verifyForgotPasswordTokenController } from '~/controllers/users.controllers' import { filterMiddleware } from '~/middlewares/common.middlewares' +import { paginationNavigator } from '~/middlewares/tweets.middlewares' import { accessTokenValidator, changePasswordValidator, @@ -190,14 +193,29 @@ usersRouter.patch( ]), wrapRequestHandler(updateMeController) ) - /** - * Description: Get User Profile - * Path: /:username + * Description: Get For you Random User By Highest Followers + * Path: /random * Method: GET + * Header: + * Body: * **/ -usersRouter.get('/:username', wrapRequestHandler(getProfileController)) +usersRouter.get('/random', wrapRequestHandler(getRandomUserController)) +/** + * Description: Get Followers + * Path: /followers + * Method: GET + * Header:{Authorization:Bearer } + * Body: + * **/ +usersRouter.get( + '/followers', + accessTokenValidator, + verifiedUSerValidator, + paginationNavigator, + wrapRequestHandler(getFollowerController) +) /** * Description: Follow Someone * Path: /follow @@ -205,6 +223,7 @@ usersRouter.get('/:username', wrapRequestHandler(getProfileController)) * Header:{Authorization:Bearer } * Body:{follower_user_id:string} * **/ + usersRouter.post( '/follow', accessTokenValidator, @@ -242,4 +261,10 @@ usersRouter.put( changePasswordValidator, wrapRequestHandler(changePasswordController) ) +/** + * Description: Get User Profile + * Path: /:username + * Method: GET + * **/ +usersRouter.get('/:username', wrapRequestHandler(getProfileController)) export default usersRouter diff --git a/src/services/bookmarks.services.ts b/src/services/bookmarks.services.ts index dd91e64..8677ea9 100644 --- a/src/services/bookmarks.services.ts +++ b/src/services/bookmarks.services.ts @@ -2,6 +2,7 @@ import Bookmark from '~/models/schemas/Bookmarks.schema' import databaseService from './database.services' import { ObjectId } from 'mongodb' import { omit } from 'lodash' +import { TweetAudience } from '~/constants/enums' class BookmarkService { async bookmarkTweet(user_id: string, tweet_id: string) { @@ -39,6 +40,292 @@ class BookmarkService { }) return result } + async getBookmarks({ user_id, page, limit }: { user_id: string; page: number; limit: number }) { + const user_id_obj = new ObjectId(user_id) + const [tweets, total] = await Promise.all([ + databaseService.bookmarks + .aggregate([ + { + $match: { + user_id: user_id_obj + } + }, + { + $lookup: { + from: 'tweets', + localField: 'tweet_id', + foreignField: '_id', + as: 'tweet' + } + }, + { + $unwind: { + path: '$tweet' + } + }, + { + $replaceRoot: { + newRoot: '$tweet' + } + }, + { + $lookup: { + from: 'users', + localField: 'user_id', + foreignField: '_id', + as: 'user' + } + }, + { + $unwind: { + path: '$user' + } + }, + { + $match: { + $or: [ + { + audience: TweetAudience.Everyone + }, + { + $and: [ + { + audience: TweetAudience.TwitterCircle + }, + { + $or: [ + { + user_id: user_id_obj + }, + { + 'user.twitter_circle': { + $in: [user_id_obj] + } + } + ] + } + ] + } + ] + } + }, + { + $lookup: { + from: 'hashtags', + localField: 'hashtags', + foreignField: '_id', + as: 'hashtags' + } + }, + { + $lookup: { + from: 'users', + localField: 'mentions', + foreignField: '_id', + as: 'mentions' + } + }, + { + $addFields: { + mentions: { + $map: { + input: '$mentions', + as: 'mention', + in: { + _id: '$$mention._id', + name: '$$mention.name', + username: '$$mention.username', + email: '$$mention.email' + } + } + } + } + }, + { + $lookup: { + from: 'bookmarks', + localField: '_id', + foreignField: 'tweet_id', + as: 'bookmarks' + } + }, + { + $lookup: { + from: 'likes', + localField: '_id', + foreignField: 'tweet_id', + as: 'likes' + } + }, + { + $lookup: { + from: 'tweets', + localField: '_id', + foreignField: 'parent_id', + as: 'tweet_children' + } + }, + { + $addFields: { + bookmarks: { + $size: '$bookmarks' + }, + likes: { + $size: '$likes' + }, + retweet_count: { + $size: { + $filter: { + input: '$tweet_children', + as: 'tweet', + cond: { + $eq: ['$$tweet.type', 1] + } + } + } + }, + comment_count: { + $size: { + $filter: { + input: '$tweet_children', + as: 'tweet', + cond: { + $eq: ['$$tweet.type', 2] + } + } + } + }, + quote_count: { + $size: { + $filter: { + input: '$tweet_children', + as: 'tweet', + cond: { + $eq: ['$$tweet.type', 3] + } + } + } + }, + total_view: { + $add: ['$user_views', '$guest_views'] + } + } + }, + { + $project: { + tweet_children: 0, + user: { + password: 0, + email_verify_token: 0, + forgot_password_token: 0, + twitter_circle: 0, + date_of_birth: 0 + } + } + }, + { + $skip: limit * (page - 1) + }, + { + $limit: limit + } + ]) + .toArray(), + databaseService.bookmarks + .aggregate([ + { + $match: { + user_id: user_id_obj + } + }, + { + $lookup: { + from: 'tweets', + localField: 'tweet_id', + foreignField: '_id', + as: 'tweet' + } + }, + { + $unwind: { + path: '$tweet' + } + }, + { + $replaceRoot: { + newRoot: '$tweet' + } + }, + { + $lookup: { + from: 'users', + localField: 'user_id', + foreignField: '_id', + as: 'user' + } + }, + { + $unwind: { + path: '$user' + } + }, + { + $match: { + $or: [ + { + audience: TweetAudience.Everyone + }, + { + $and: [ + { + audience: TweetAudience.TwitterCircle + }, + { + $or: [ + { + user_id: user_id_obj + }, + { + 'user.twitter_circle': { + $in: [user_id_obj] + } + } + ] + } + ] + } + ] + } + } + ]) + .toArray() + ]) + + const tweet_ids = tweets.map((tweet) => tweet._id as ObjectId) + const date = new Date() + await databaseService.tweets.updateMany( + { + _id: { + $in: tweet_ids + } + }, + { + $inc: { user_views: 1 }, + $set: { + updated_at: date + } + } + ) + tweets.forEach((tweet) => { + tweet.updated_at = date + tweet.user_views += 1 + }) + + return { + tweets: tweets, + total: total ? total?.length : 0 + } + } } const bookmarkService = new BookmarkService() export default bookmarkService diff --git a/src/services/database.services.ts b/src/services/database.services.ts index 1f5cfef..468bf01 100644 --- a/src/services/database.services.ts +++ b/src/services/database.services.ts @@ -9,6 +9,7 @@ import Bookmark from '~/models/schemas/Bookmarks.schema' import Like from '~/models/schemas/Likes.schema' import Conversation from '~/models/schemas/Conversations.schema' import { envConfig } from '~/constants/config' +import Notification from '~/models/schemas/Notifications.schema' const uri = `mongodb+srv://${envConfig.dbUsername}:${envConfig.dbPassword}@twitter.edfvckz.mongodb.net/?retryWrites=true&w=majority` class DatabaseService { @@ -135,6 +136,9 @@ class DatabaseService { get conversations(): Collection { return this.db.collection(envConfig.dbConversationCollection as string) } + get notifications(): Collection { + return this.db.collection(envConfig.dbNotificationCollection as string) + } } // Tạo Object từ class DatabaseService diff --git a/src/services/likes.services.ts b/src/services/likes.services.ts index 0815c40..8940423 100644 --- a/src/services/likes.services.ts +++ b/src/services/likes.services.ts @@ -1,6 +1,7 @@ import { ObjectId } from 'mongodb' import databaseService from './database.services' import Like from '~/models/schemas/Likes.schema' +import { TweetAudience } from '~/constants/enums' class LikeService { async likeTweet(user_id: string, tweet_id: string) { @@ -35,6 +36,292 @@ class LikeService { }) return result } + async getTweetUserLiked({ user_id, page, limit }: { user_id: string; page: number; limit: number }) { + const user_id_obj = new ObjectId(user_id) + const [tweets, total] = await Promise.all([ + databaseService.likes + .aggregate([ + { + $match: { + user_id: user_id_obj + } + }, + { + $lookup: { + from: 'tweets', + localField: 'tweet_id', + foreignField: '_id', + as: 'tweet' + } + }, + { + $unwind: { + path: '$tweet' + } + }, + { + $replaceRoot: { + newRoot: '$tweet' + } + }, + { + $lookup: { + from: 'users', + localField: 'user_id', + foreignField: '_id', + as: 'user' + } + }, + { + $unwind: { + path: '$user' + } + }, + { + $match: { + $or: [ + { + audience: TweetAudience.Everyone + }, + { + $and: [ + { + audience: TweetAudience.TwitterCircle + }, + { + $or: [ + { + user_id: user_id_obj + }, + { + 'user.twitter_circle': { + $in: [user_id_obj] + } + } + ] + } + ] + } + ] + } + }, + { + $lookup: { + from: 'hashtags', + localField: 'hashtags', + foreignField: '_id', + as: 'hashtags' + } + }, + { + $lookup: { + from: 'users', + localField: 'mentions', + foreignField: '_id', + as: 'mentions' + } + }, + { + $addFields: { + mentions: { + $map: { + input: '$mentions', + as: 'mention', + in: { + _id: '$$mention._id', + name: '$$mention.name', + username: '$$mention.username', + email: '$$mention.email' + } + } + } + } + }, + { + $lookup: { + from: 'bookmarks', + localField: '_id', + foreignField: 'tweet_id', + as: 'bookmarks' + } + }, + { + $lookup: { + from: 'likes', + localField: '_id', + foreignField: 'tweet_id', + as: 'likes' + } + }, + { + $lookup: { + from: 'tweets', + localField: '_id', + foreignField: 'parent_id', + as: 'tweet_children' + } + }, + { + $addFields: { + bookmarks: { + $size: '$bookmarks' + }, + likes: { + $size: '$likes' + }, + retweet_count: { + $size: { + $filter: { + input: '$tweet_children', + as: 'tweet', + cond: { + $eq: ['$$tweet.type', 1] + } + } + } + }, + comment_count: { + $size: { + $filter: { + input: '$tweet_children', + as: 'tweet', + cond: { + $eq: ['$$tweet.type', 2] + } + } + } + }, + quote_count: { + $size: { + $filter: { + input: '$tweet_children', + as: 'tweet', + cond: { + $eq: ['$$tweet.type', 3] + } + } + } + }, + total_view: { + $add: ['$user_views', '$guest_views'] + } + } + }, + { + $project: { + tweet_children: 0, + user: { + password: 0, + email_verify_token: 0, + forgot_password_token: 0, + twitter_circle: 0, + date_of_birth: 0 + } + } + }, + { + $skip: limit * (page - 1) + }, + { + $limit: limit + } + ]) + .toArray(), + databaseService.bookmarks + .aggregate([ + { + $match: { + user_id: user_id_obj + } + }, + { + $lookup: { + from: 'tweets', + localField: 'tweet_id', + foreignField: '_id', + as: 'tweet' + } + }, + { + $unwind: { + path: '$tweet' + } + }, + { + $replaceRoot: { + newRoot: '$tweet' + } + }, + { + $lookup: { + from: 'users', + localField: 'user_id', + foreignField: '_id', + as: 'user' + } + }, + { + $unwind: { + path: '$user' + } + }, + { + $match: { + $or: [ + { + audience: TweetAudience.Everyone + }, + { + $and: [ + { + audience: TweetAudience.TwitterCircle + }, + { + $or: [ + { + user_id: user_id_obj + }, + { + 'user.twitter_circle': { + $in: [user_id_obj] + } + } + ] + } + ] + } + ] + } + } + ]) + .toArray() + ]) + + const tweet_ids = tweets.map((tweet) => tweet._id as ObjectId) + const date = new Date() + await databaseService.tweets.updateMany( + { + _id: { + $in: tweet_ids + } + }, + { + $inc: { user_views: 1 }, + $set: { + updated_at: date + } + } + ) + tweets.forEach((tweet) => { + tweet.updated_at = date + tweet.user_views += 1 + }) + + return { + tweets: tweets, + total: total ? total?.length : 0 + } + } } const likeService = new LikeService() diff --git a/src/services/notifications.services.ts b/src/services/notifications.services.ts new file mode 100644 index 0000000..a191cff --- /dev/null +++ b/src/services/notifications.services.ts @@ -0,0 +1,22 @@ +import { ObjectId } from 'mongodb' +import { NotificationRequestBody } from '~/models/requests/Notification.requests' +import Notification from '~/models/schemas/Notifications.schema' +import databaseService from './database.services' + +class NotificationService { + async createNotification(user_id: string, body: NotificationRequestBody) { + const { message, owner_username, interacted_username, tweet_id, children_tweet } = body + const notification = new Notification({ + message, + owner_username, + interacted_username, + tweet_id: new ObjectId(tweet_id), + children_tweet: children_tweet ? new ObjectId(children_tweet) : null + }) + const result = await databaseService.notifications.insertOne(notification) + return result + } +} + +const notificationService = new NotificationService() +export default notificationService diff --git a/src/services/tweets.services.ts b/src/services/tweets.services.ts index cf10651..2a12daa 100644 --- a/src/services/tweets.services.ts +++ b/src/services/tweets.services.ts @@ -4,6 +4,7 @@ import Tweet from '~/models/schemas/Tweet.schema' import { ObjectId, WithId } from 'mongodb' import Hashtag from '~/models/schemas/Hashtag.schema' import { TweetType } from '~/constants/enums' +import usersService from './users.services' class TweetsService { async checkAndCreateHashtags(hashtags: string[]) { @@ -235,7 +236,17 @@ class TweetsService { }) return { tweets, total } } - async getNewFeeds({ user_id, limit, page }: { user_id: string; limit: number; page: number }) { + async getNewFeeds({ + user_id, + limit, + page, + isForYou + }: { + user_id: string + limit: number + page: number + isForYou?: boolean + }) { const user_id_obj = new ObjectId(user_id) const followed_user_ids = await databaseService.followers .find( @@ -250,7 +261,14 @@ class TweetsService { } ) .toArray() - const ids = followed_user_ids.map((followed_user_id) => followed_user_id.followed_user_id) + + let ids = followed_user_ids.map((followed_user_id) => followed_user_id.followed_user_id) + if (isForYou) { + const { unFollow } = await usersService.getUnFollower(user_id) + + const not_following = unFollow.map((unFollow_userID: any) => unFollow_userID._id) + ids = [...not_following, ids] + } //Mong muốn newfeed sẽ lấy luôn feed của mình ids.push(new ObjectId(user_id)) @@ -424,6 +442,7 @@ class TweetsService { } ]) .toArray(), + databaseService.tweets .aggregate([ { diff --git a/src/services/users.services.ts b/src/services/users.services.ts index ec8f7df..dbe1c37 100644 --- a/src/services/users.services.ts +++ b/src/services/users.services.ts @@ -468,6 +468,197 @@ class UsersService { message: USERS_MESSAGES.CHANGE_PASSWORDS_SUCCESS } } + + async getRandomUser() { + const users_random = await databaseService.users + .aggregate([ + { + $lookup: { + from: 'followers', + localField: '_id', + foreignField: 'followed_user_id', + as: 'followers' + } + }, + { + $addFields: { + followers: { + $size: '$followers' + } + } + }, + { + $project: { + _id: 1, + name: 1, + username: 1, + bio: 1, + followers: 1, + avatar: 1, + cover_photo: 1 + } + }, + { + $sort: { + followers: -1 + } + }, + { + $limit: 10 + } + ]) + .toArray() + + return users_random + } + async getFollower({ user_id, page, limit }: { user_id: string; page: number; limit: number }) { + const user_id_obj = new ObjectId(user_id) + + const [followers, total] = await Promise.all([ + databaseService.users + .aggregate([ + { + $match: { + _id: user_id_obj + } + }, + { + $lookup: { + from: 'followers', + localField: '_id', + foreignField: 'followed_user_id', + as: 'followers' + } + }, + { + $unwind: { + path: '$followers' + } + }, + { + $replaceRoot: { + newRoot: '$followers' + } + }, + { + $limit: limit + }, + { + $skip: limit * (page - 1) + } + ]) + .toArray(), + databaseService.users + .aggregate([ + { + $match: { + _id: user_id_obj + } + }, + { + $lookup: { + from: 'followers', + localField: '_id', + foreignField: 'followed_user_id', + as: 'followers' + } + }, + { + $addFields: + /** + * newField: The new field name. + * expression: The new field expression. + */ + { + total_followers: { + $size: '$followers' + } + } + } + ]) + .toArray() + ]) + return { + followers: followers, + total: total[0].total_followers + } + } + async getUnFollower(user_id: string, limit: number = 10) { + const user_id_obj = new ObjectId(user_id) + const user = await databaseService.users + .aggregate([ + { + $match: { + _id: user_id_obj + } + }, + { + $lookup: { + from: 'followers', + localField: '_id', + foreignField: 'followed_user_id', + as: 'followers' + } + }, + { + $addFields: { + follower_ids: { + $map: { + input: '$followers', + as: 'follower', + in: { + _id: '$$follower.user_id' + } + } + } + } + }, + { + $lookup: { + from: 'users', + let: { + user_id: '$_id', + follower_ids: '$follower_ids' + }, + pipeline: [ + { + $match: { + $expr: { + $and: [ + { + $not: { + $in: ['$_id', '$$follower_ids'] + } + } + ] + } + } + }, + { + $project: { + _id: 1 + } + } + ], + as: 'not_following' + } + }, + { + $project: { + not_following: 1, + _id: 0 + } + }, + { + $limit: limit + } + ]) + .toArray() + + return { + unFollow: (user[0] as any)?.not_following + } + } } const usersService = new UsersService() diff --git a/src/utils/socket.ts b/src/utils/socket.ts index 74b1124..bb80bc9 100644 --- a/src/utils/socket.ts +++ b/src/utils/socket.ts @@ -84,7 +84,6 @@ export const initSocket = (httpServer: HTTP_SERVER.Server) => { }) } }) - socket.on('disconnect', () => { users.delete(user_id) // delete users[user_id]