diff --git a/.github/workflows/pull_changes.yml b/.github/workflows/pull_changes.yml new file mode 100644 index 0000000000..006cd1ad25 --- /dev/null +++ b/.github/workflows/pull_changes.yml @@ -0,0 +1,23 @@ +name: Copy Docs to Talawa Docs + +on: + schedule: + - cron: '*/5 * * * *' +jobs: + copy-docs-to-talawa-docs: + runs-on: ubuntu-latest + if: github.ref == 'refs/heads/automated-docs' + # needs: Generate-Documentation + steps: + - uses: actions/checkout@v3 + - uses: dmnemec/copy_file_to_another_repo_action@v1.1.1 + env: + API_TOKEN_GITHUB: ${{secrets.TALAWA_DOCS_SYNC}} + with: + source_file: 'talawa-api-docs/' + destination_repo: 'PalisadoesFoundation/talawa-docs' + destination_branch: 'develop' + destination_folder: 'docs/' + user_email: '${{env.email}}' + user_name: '${{github.actor}}' + commit_message: 'Talawa API docs updated' diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index 6eb103c38b..facb5bc15f 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -96,7 +96,7 @@ jobs: Generate-Documentation: name: Generate Documentation runs-on: ubuntu-latest - # if: github.ref == 'refs/heads/automated-docs' + if: github.ref == 'refs/heads/develop' needs: Push-Workflow steps: - name: Checkout repository @@ -150,23 +150,24 @@ jobs: name: documentation-api path: talawa-api-docs - Copy-docs-to-talawa-docs: - runs-on: ubuntu-latest - needs: Generate-Documentation - steps: - - uses: actions/checkout@v3 - - uses: dmnemec/copy_file_to_another_repo_action@v1.1.1 - env: - API_TOKEN_GITHUB: ${{secrets.TALAWA_DOCS_SYNC}} - with: - source_file: 'talawa-api-docs/' - destination_repo: 'PalisadoesFoundation/talawa-docs' - destination_branch: 'develop' - destination_folder: 'docs/' - user_email: '${{env.email}}' - user_name: '${{github.actor}}' - commit_message: 'Talawa API docs updated' - + # Copy-docs-to-talawa-docs: + # runs-on: ubuntu-latest + # if: github.ref == 'refs/heads/automated-docs' + # # needs: Generate-Documentation + # steps: + # - uses: actions/checkout@v3 + # - uses: dmnemec/copy_file_to_another_repo_action@v1.1.1 + # env: + # API_TOKEN_GITHUB: ${{secrets.TALAWA_DOCS_SYNC}} + # with: + # source_file: 'talawa-api-docs/' + # destination_repo: 'PalisadoesFoundation/talawa-docs' + # destination_branch: 'develop' + # destination_folder: 'docs/' + # user_email: '${{env.email}}' + # user_name: '${{github.actor}}' + # commit_message: 'Talawa API docs updated' + # You can find the deployment instructions in the scripts/cloud-api-demo/README.md file Deploy-Workflow: name: Deploying Application to Cloud VPS diff --git a/INSTALLATION.md b/INSTALLATION.md index 5aafbc4ba2..1e8e36b17e 100644 --- a/INSTALLATION.md +++ b/INSTALLATION.md @@ -76,14 +76,16 @@ You will need to have copies of your code on your local system. Here's how to do ## Install node.js -Best way to install and manage `node.js` is making use of node version managers. Two most popular node version managers right now are [fnm](https://github.com/Schniz/fnm) and [nvm](https://github.com/nvm-sh/nvm). We'd recommend `fnm` because it's written in `rust` and is much faster than `nvm`. Install whichever one you want and follow their guide to set up `node.js` on your system ensure the installation of Node.js version 20 LTS. +Best way to install and manage `node.js` is making use of node version managers. Two most popular node version managers right now are [fnm](https://github.com/Schniz/fnm) and [nvm](https://github.com/nvm-sh/nvm). We'd recommend `fnm` because it's written in `rust` and is much faster than `nvm`. Install whichever one you want and follow their guide to set up `node.js` on your system ensure the installation of Node.js version 20 LTS. ## Install npm npm is a package manager for Node.js and is installed with Node.js. npm is used to install, share, and distribute code as well as to manage dependencies in your projects. To check if you have npm installed you can run this command in your terminal: + ``` npm -v ``` + If you have it installed then you should see the version that's installed. If not, you can download Node.js and npm from the official [Node.js website](https://nodejs.org/en/download/). ## Install TypeScript @@ -95,6 +97,7 @@ To install TypeScript, you can use npm: ```bash npm install -g typescript ``` + This command installs TypeScript globally on your system so that it can be accessed from any project. ## Install git @@ -250,7 +253,7 @@ Remember to adjust any paths or details as needed for your specific environment. It's important to configure Talawa-API to complete it's setup. -A configuration file named `.env` is required in the root directory of `Talawa-API` for storing environment variables used at runtime. It is not a part of the repo and you will have to create it. +A configuration file named `.env` is required in the root directory of `Talawa-API` for storing environment variables used at runtime. It is not a part of the repo and you will have to create it. ## Automated Configuration of `.env` @@ -275,27 +278,28 @@ Use this command to do this ``` cp .env.sample .env ``` + ### The Environment Variables in `.env` This `.env` file must be populated with the following environment variables for talawa-api to work: -| Variable | Description | -| ---------------------------- | ------------------------------------------------------ | -| NODE_ENV | Used for providing the environment in which the the talawa-api is running | -| ACCESS_TOKEN_SECRET | Used for signing/verifying JWT tokens | -| REFRESH_TOKEN_SECRET | Used for signing/verifying JWT tokens | -| MONGO_DB_URL | Used for connecting talawa-api to the mongoDB database | -| RECAPTCHA_SECRET_KEY | Used for authentication using reCAPTCHA | -| RECAPTCHA_SITE_KEY | Used for authentication using reCAPTCHA | -| MAIL_USERNAME | Used for mailing service | -| MAIL_PASSWORD | Used for mailing service | -| LAST_RESORT_SUPERADMIN_EMAIL | Used for promoting the default super admin | -| COLORIZE_LOGS | Used for colorized log formats in console | -| LOG_LEVEL | Used for setting the logging level | -| REDIS HOST | Used for connecting talawa-api to the redis instance | -| REDIS_PORT | Specifies the port of the active redis-server | -| REDIS_PASSWORD(optional) | Used for authenticating the connection request to | -| | a hosted redis-server | +| Variable | Description | +| ---------------------------- | ------------------------------------------------------------------------- | +| NODE_ENV | Used for providing the environment in which the the talawa-api is running | +| ACCESS_TOKEN_SECRET | Used for signing/verifying JWT tokens | +| REFRESH_TOKEN_SECRET | Used for signing/verifying JWT tokens | +| MONGO_DB_URL | Used for connecting talawa-api to the mongoDB database | +| RECAPTCHA_SECRET_KEY | Used for authentication using reCAPTCHA | +| RECAPTCHA_SITE_KEY | Used for authentication using reCAPTCHA | +| MAIL_USERNAME | Used for mailing service | +| MAIL_PASSWORD | Used for mailing service | +| LAST_RESORT_SUPERADMIN_EMAIL | Used for promoting the default super admin | +| COLORIZE_LOGS | Used for colorized log formats in console | +| LOG_LEVEL | Used for setting the logging level | +| REDIS HOST | Used for connecting talawa-api to the redis instance | +| REDIS_PORT | Specifies the port of the active redis-server | +| REDIS_PASSWORD(optional) | Used for authenticating the connection request to | +| | a hosted redis-server | The following sections will show you how to configure each of these parameters. @@ -532,7 +536,6 @@ If the parameter value is set to `true`, you should be able to see colorized log There are different logging levels that can be configured by setting this parameter. The severity order of levels are displayed numerically ascending from most important to least important. - ``` levels = { error: 0, @@ -571,8 +574,9 @@ You can pass the following arguments while running this script. - `npm run import:sample-data`: This command will import the complete sample database without removing the existing data. - `npm run import:sample-data -- --format`: This command will import the complete sample database after removing the existing data. -- `npm run import:sample-data -- --format --items=users,organizations`: This command will import the sample `users` and `organizations` collections after cleaning the existing data. -- `npm run import:sample-data -- --items=users,organizations`: This command will import the sample `users` and `organizations` collections without cleaning the existing data. +- `npm run import:sample-data -- --format --items=users,organizations,appUserProfiles`: This command will import the sample `users` , `organizations` and `appUserProfiles` collections after cleaning the existing data. +- `npm run import:sample-data -- --items=users,organizations,appUserProfiles`: This command will import the sample `users` , `organizations` + ans `appUserProfiles` collections without cleaning the existing data. ## Sample Data Overview: @@ -769,4 +773,4 @@ Talawa-api makes use of `vitest` to run tests because it is much faster than `je You can run the tests for talawa-api using this command: - npm run test \ No newline at end of file + npm run test diff --git a/README.md b/README.md index 327fa7941a..200e9cffa8 100644 --- a/README.md +++ b/README.md @@ -25,11 +25,10 @@ Core features include: -- [Talawa API](#talawa-api) - - [Talawa Components](#talawa-components) - - [Documentation](#documentation) - - [Installation](#installation) - - [Image Upload](#image-upload) +- [Talawa Components](#talawa-components) +- [Documentation](#documentation) +- [Installation](#installation) +- [Image Upload](#image-upload) diff --git a/package-lock.json b/package-lock.json index f7aecf2450..12919106f9 100644 --- a/package-lock.json +++ b/package-lock.json @@ -49,7 +49,7 @@ "mongoose-paginate-v2": "^1.7.0", "morgan": "^1.10.0", "nanoid": "^5.0.5", - "nodemailer": "^6.9.8", + "nodemailer": "^6.9.9", "pm2": "^5.2.0", "redis": "^4.6.12", "shortid": "^2.2.16", @@ -64,7 +64,7 @@ }, "devDependencies": { "@graphql-codegen/cli": "^5.0.0", - "@graphql-codegen/typescript": "^4.0.1", + "@graphql-codegen/typescript": "^4.0.4", "@graphql-codegen/typescript-resolvers": "^4.0.1", "@graphql-eslint/eslint-plugin": "^3.20.1", "@types/bcryptjs": "^2.4.6", @@ -84,8 +84,8 @@ "@types/nodemailer": "^6.4.14", "@types/shortid": "^0.0.32", "@types/uuid": "^9.0.7", - "@types/validator": "^13.11.7", - "@typescript-eslint/eslint-plugin": "^6.20.0", + "@types/validator": "^13.11.9", + "@typescript-eslint/eslint-plugin": "^6.21.0", "@typescript-eslint/parser": "^6.20.0", "@vitest/coverage-c8": "^0.24.3", "concurrently": "^8.2.2", @@ -96,7 +96,7 @@ "get-graphql-schema": "^2.1.2", "graphql-markdown": "^7.0.0", "husky": "^9.0.6", - "lint-staged": "^15.2.1", + "lint-staged": "^15.2.2", "prettier": "^2.7.1", "rimraf": "^5.0.5", "tsx": "^4.7.0", @@ -2640,9 +2640,9 @@ "dev": true }, "node_modules/@graphql-codegen/plugin-helpers": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/@graphql-codegen/plugin-helpers/-/plugin-helpers-5.0.1.tgz", - "integrity": "sha512-6L5sb9D8wptZhnhLLBcheSPU7Tg//DGWgc5tQBWX46KYTOTQHGqDpv50FxAJJOyFVJrveN9otWk9UT9/yfY4ww==", + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/@graphql-codegen/plugin-helpers/-/plugin-helpers-5.0.3.tgz", + "integrity": "sha512-yZ1rpULIWKBZqCDlvGIJRSyj1B2utkEdGmXZTBT/GVayP4hyRYlkd36AJV/LfEsVD8dnsKL5rLz2VTYmRNlJ5Q==", "dev": true, "dependencies": { "@graphql-tools/utils": "^10.0.0", @@ -2650,49 +2650,37 @@ "common-tags": "1.8.2", "import-from": "4.0.0", "lodash": "~4.17.0", - "tslib": "~2.5.0" + "tslib": "~2.6.0" }, "peerDependencies": { "graphql": "^0.8.0 || ^0.9.0 || ^0.10.0 || ^0.11.0 || ^0.12.0 || ^0.13.0 || ^14.0.0 || ^15.0.0 || ^16.0.0" } }, - "node_modules/@graphql-codegen/plugin-helpers/node_modules/tslib": { - "version": "2.5.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.3.tgz", - "integrity": "sha512-mSxlJJwl3BMEQCUNnxXBU9jP4JBktcEGhURcPR6VQVlnP0FdDEsIaz0C35dXNGLyRfrATNofF0F5p2KPxQgB+w==", - "dev": true - }, "node_modules/@graphql-codegen/schema-ast": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/@graphql-codegen/schema-ast/-/schema-ast-4.0.0.tgz", - "integrity": "sha512-WIzkJFa9Gz28FITAPILbt+7A8+yzOyd1NxgwFh7ie+EmO9a5zQK6UQ3U/BviirguXCYnn+AR4dXsoDrSrtRA1g==", + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@graphql-codegen/schema-ast/-/schema-ast-4.0.2.tgz", + "integrity": "sha512-5mVAOQQK3Oz7EtMl/l3vOQdc2aYClUzVDHHkMvZlunc+KlGgl81j8TLa+X7ANIllqU4fUEsQU3lJmk4hXP6K7Q==", "dev": true, "dependencies": { - "@graphql-codegen/plugin-helpers": "^5.0.0", + "@graphql-codegen/plugin-helpers": "^5.0.3", "@graphql-tools/utils": "^10.0.0", - "tslib": "~2.5.0" + "tslib": "~2.6.0" }, "peerDependencies": { "graphql": "^0.8.0 || ^0.9.0 || ^0.10.0 || ^0.11.0 || ^0.12.0 || ^0.13.0 || ^14.0.0 || ^15.0.0 || ^16.0.0" } }, - "node_modules/@graphql-codegen/schema-ast/node_modules/tslib": { - "version": "2.5.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.3.tgz", - "integrity": "sha512-mSxlJJwl3BMEQCUNnxXBU9jP4JBktcEGhURcPR6VQVlnP0FdDEsIaz0C35dXNGLyRfrATNofF0F5p2KPxQgB+w==", - "dev": true - }, "node_modules/@graphql-codegen/typescript": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/@graphql-codegen/typescript/-/typescript-4.0.1.tgz", - "integrity": "sha512-3YziQ21dCVdnHb+Us1uDb3pA6eG5Chjv0uTK+bt9dXeMlwYBU8MbtzvQTo4qvzWVC1AxSOKj0rgfNu1xCXqJyA==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/@graphql-codegen/typescript/-/typescript-4.0.4.tgz", + "integrity": "sha512-x79CKLfP9UQCX+/I78qxQlMs2Mmq3pF1lKafZo7lAno0f/fvJ+qWUduzdgjRNz+YL+5blGeWcC0pWEDxniO7hw==", "dev": true, "dependencies": { - "@graphql-codegen/plugin-helpers": "^5.0.0", - "@graphql-codegen/schema-ast": "^4.0.0", - "@graphql-codegen/visitor-plugin-common": "4.0.1", + "@graphql-codegen/plugin-helpers": "^5.0.3", + "@graphql-codegen/schema-ast": "^4.0.2", + "@graphql-codegen/visitor-plugin-common": "4.1.2", "auto-bind": "~4.0.0", - "tslib": "~2.5.0" + "tslib": "~2.6.0" }, "peerDependencies": { "graphql": "^0.12.0 || ^0.13.0 || ^14.0.0 || ^15.0.0 || ^16.0.0" @@ -2721,11 +2709,26 @@ "integrity": "sha512-mSxlJJwl3BMEQCUNnxXBU9jP4JBktcEGhURcPR6VQVlnP0FdDEsIaz0C35dXNGLyRfrATNofF0F5p2KPxQgB+w==", "dev": true }, - "node_modules/@graphql-codegen/typescript/node_modules/tslib": { - "version": "2.5.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.3.tgz", - "integrity": "sha512-mSxlJJwl3BMEQCUNnxXBU9jP4JBktcEGhURcPR6VQVlnP0FdDEsIaz0C35dXNGLyRfrATNofF0F5p2KPxQgB+w==", - "dev": true + "node_modules/@graphql-codegen/typescript/node_modules/@graphql-codegen/visitor-plugin-common": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/@graphql-codegen/visitor-plugin-common/-/visitor-plugin-common-4.1.2.tgz", + "integrity": "sha512-yk7iEAL1kYZ2Gi/pvVjdsZhul5WsYEM4Zcgh2Ev15VicMdJmPHsMhNUsZWyVJV0CaQCYpNOFlGD/11Ea3pn4GA==", + "dev": true, + "dependencies": { + "@graphql-codegen/plugin-helpers": "^5.0.3", + "@graphql-tools/optimize": "^2.0.0", + "@graphql-tools/relay-operation-optimizer": "^7.0.0", + "@graphql-tools/utils": "^10.0.0", + "auto-bind": "~4.0.0", + "change-case-all": "1.0.15", + "dependency-graph": "^0.11.0", + "graphql-tag": "^2.11.0", + "parse-filepath": "^1.0.2", + "tslib": "~2.6.0" + }, + "peerDependencies": { + "graphql": "^0.8.0 || ^0.9.0 || ^0.10.0 || ^0.11.0 || ^0.12.0 || ^0.13.0 || ^14.0.0 || ^15.0.0 || ^16.0.0" + } }, "node_modules/@graphql-codegen/visitor-plugin-common": { "version": "4.0.1", @@ -6168,9 +6171,9 @@ "integrity": "sha512-WZLiwShhwLRmeV6zH+GkbOFT6Z6VklCItrDioxUnv+u4Ll+8vKeFySoFyK/0ctcRpOmwAicELfmys1sDc/Rw+A==" }, "node_modules/@types/semver": { - "version": "7.5.6", - "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.6.tgz", - "integrity": "sha512-dn1l8LaMea/IjDoHNd9J52uBbInB796CDffS6VdIxvqYCPSG0V0DzHp76GpaWnlhg88uYyPbXCDIowa86ybd5A==", + "version": "7.5.7", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.7.tgz", + "integrity": "sha512-/wdoPq1QqkSj9/QOeKkFquEuPzQbHTWAMPH/PaUMB+JuR31lXhlWXRZ52IpfDYVlDOUBvX09uBrPwxGT1hjNBg==", "dev": true }, "node_modules/@types/send": { @@ -6219,9 +6222,9 @@ "dev": true }, "node_modules/@types/validator": { - "version": "13.11.8", - "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.11.8.tgz", - "integrity": "sha512-c/hzNDBh7eRF+KbCf+OoZxKbnkpaK/cKp9iLQWqB7muXtM+MtL9SUUH8vCFcLn6dH1Qm05jiexK0ofWY7TfOhQ==", + "version": "13.11.9", + "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.11.9.tgz", + "integrity": "sha512-FCTsikRozryfayPuiI46QzH3fnrOoctTjvOYZkho9BTFLCOZ2rgZJHMOVgCOfttjPJcgOx52EpkY0CMfy87MIw==", "dev": true }, "node_modules/@types/ws": { @@ -6246,16 +6249,16 @@ "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==" }, "node_modules/@typescript-eslint/eslint-plugin": { - "version": "6.20.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.20.0.tgz", - "integrity": "sha512-fTwGQUnjhoYHeSF6m5pWNkzmDDdsKELYrOBxhjMrofPqCkoC2k3B2wvGHFxa1CTIqkEn88nlW1HVMztjo2K8Hg==", + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.21.0.tgz", + "integrity": "sha512-oy9+hTPCUFpngkEZUSzbf9MxI65wbKFoQYsgPdILTfbUldp5ovUuphZVe4i30emU9M/kP+T64Di0mxl7dSw3MA==", "dev": true, "dependencies": { "@eslint-community/regexpp": "^4.5.1", - "@typescript-eslint/scope-manager": "6.20.0", - "@typescript-eslint/type-utils": "6.20.0", - "@typescript-eslint/utils": "6.20.0", - "@typescript-eslint/visitor-keys": "6.20.0", + "@typescript-eslint/scope-manager": "6.21.0", + "@typescript-eslint/type-utils": "6.21.0", + "@typescript-eslint/utils": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", "debug": "^4.3.4", "graphemer": "^1.4.0", "ignore": "^5.2.4", @@ -6280,6 +6283,53 @@ } } }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/scope-manager": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.21.0.tgz", + "integrity": "sha512-OwLUIWZJry80O99zvqXVEioyniJMa+d2GrqpUTqi5/v5D5rOrppJVBPa0yKCblcigC0/aYAzxxqQ1B+DS2RYsg==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", + "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", + "dev": true, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/visitor-keys": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", + "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, "node_modules/@typescript-eslint/eslint-plugin/node_modules/lru-cache": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", @@ -6353,13 +6403,13 @@ } }, "node_modules/@typescript-eslint/type-utils": { - "version": "6.20.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.20.0.tgz", - "integrity": "sha512-qnSobiJQb1F5JjN0YDRPHruQTrX7ICsmltXhkV536mp4idGAYrIyr47zF/JmkJtEcAVnIz4gUYJ7gOZa6SmN4g==", + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.21.0.tgz", + "integrity": "sha512-rZQI7wHfao8qMX3Rd3xqeYSMCL3SoiSQLBATSiVKARdFGCYSRvmViieZjqc58jKgs8Y8i9YvVVhRbHSTA4VBag==", "dev": true, "dependencies": { - "@typescript-eslint/typescript-estree": "6.20.0", - "@typescript-eslint/utils": "6.20.0", + "@typescript-eslint/typescript-estree": "6.21.0", + "@typescript-eslint/utils": "6.21.0", "debug": "^4.3.4", "ts-api-utils": "^1.0.1" }, @@ -6379,6 +6429,115 @@ } } }, + "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", + "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", + "dev": true, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/typescript-estree": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.21.0.tgz", + "integrity": "sha512-6npJTkZcO+y2/kr+z0hc4HwNfrrP4kNYh57ek7yCNlrBjWQ1Y0OS7jiZTkgumrvkX5HkEKXFZkkdFNkaW2wmUQ==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "minimatch": "9.0.3", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/visitor-keys": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", + "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils/node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/@typescript-eslint/type-utils/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@typescript-eslint/type-utils/node_modules/minimatch": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", + "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "dev": true, + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/@typescript-eslint/type-utils/node_modules/semver": { + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/@typescript-eslint/types": { "version": "6.20.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.20.0.tgz", @@ -6472,17 +6631,17 @@ } }, "node_modules/@typescript-eslint/utils": { - "version": "6.20.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.20.0.tgz", - "integrity": "sha512-/EKuw+kRu2vAqCoDwDCBtDRU6CTKbUmwwI7SH7AashZ+W+7o8eiyy6V2cdOqN49KsTcASWsC5QeghYuRDTyOOg==", + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.21.0.tgz", + "integrity": "sha512-NfWVaC8HP9T8cbKQxHcsJBY5YE1O33+jpMwN45qzWWaPDZgLIbo12toGMWnmhvCpd3sIxkpDw3Wv1B3dYrbDQQ==", "dev": true, "dependencies": { "@eslint-community/eslint-utils": "^4.4.0", "@types/json-schema": "^7.0.12", "@types/semver": "^7.5.0", - "@typescript-eslint/scope-manager": "6.20.0", - "@typescript-eslint/types": "6.20.0", - "@typescript-eslint/typescript-estree": "6.20.0", + "@typescript-eslint/scope-manager": "6.21.0", + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/typescript-estree": "6.21.0", "semver": "^7.5.4" }, "engines": { @@ -6496,6 +6655,90 @@ "eslint": "^7.0.0 || ^8.0.0" } }, + "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/scope-manager": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.21.0.tgz", + "integrity": "sha512-OwLUIWZJry80O99zvqXVEioyniJMa+d2GrqpUTqi5/v5D5rOrppJVBPa0yKCblcigC0/aYAzxxqQ1B+DS2RYsg==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", + "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", + "dev": true, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/typescript-estree": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.21.0.tgz", + "integrity": "sha512-6npJTkZcO+y2/kr+z0hc4HwNfrrP4kNYh57ek7yCNlrBjWQ1Y0OS7jiZTkgumrvkX5HkEKXFZkkdFNkaW2wmUQ==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "minimatch": "9.0.3", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/visitor-keys": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", + "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/utils/node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0" + } + }, "node_modules/@typescript-eslint/utils/node_modules/lru-cache": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", @@ -6508,10 +6751,25 @@ "node": ">=10" } }, + "node_modules/@typescript-eslint/utils/node_modules/minimatch": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", + "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "dev": true, + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, "node_modules/@typescript-eslint/utils/node_modules/semver": { - "version": "7.5.4", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", - "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", "dev": true, "dependencies": { "lru-cache": "^6.0.0" @@ -12158,9 +12416,9 @@ "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==" }, "node_modules/lint-staged": { - "version": "15.2.1", - "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.2.1.tgz", - "integrity": "sha512-dhwAPnM85VdshybV9FWI/9ghTvMLoQLEXgVMx+ua2DN7mdfzd/tRfoU2yhMcBac0RHkofoxdnnJUokr8s4zKmQ==", + "version": "15.2.2", + "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.2.2.tgz", + "integrity": "sha512-TiTt93OPh1OZOsb5B7k96A/ATl2AjIZo+vnzFZ6oHK5FuTk63ByDtxGQpHm+kFETjEWqgkF95M8FRXKR/LEBcw==", "dev": true, "dependencies": { "chalk": "5.3.0", @@ -13417,9 +13675,9 @@ "integrity": "sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==" }, "node_modules/nodemailer": { - "version": "6.9.8", - "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.9.8.tgz", - "integrity": "sha512-cfrYUk16e67Ks051i4CntM9kshRYei1/o/Gi8K1d+R34OIs21xdFnW7Pt7EucmVKA0LKtqUGNcjMZ7ehjl49mQ==", + "version": "6.9.9", + "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.9.9.tgz", + "integrity": "sha512-dexTll8zqQoVJEZPwQAKzxxtFn0qTnjdQTchoU6Re9BUUGBJiOy3YMn/0ShTW6J5M0dfQ1NeDeRTTl4oIWgQMA==", "engines": { "node": ">=6.0.0" } @@ -18935,9 +19193,9 @@ } }, "@graphql-codegen/plugin-helpers": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/@graphql-codegen/plugin-helpers/-/plugin-helpers-5.0.1.tgz", - "integrity": "sha512-6L5sb9D8wptZhnhLLBcheSPU7Tg//DGWgc5tQBWX46KYTOTQHGqDpv50FxAJJOyFVJrveN9otWk9UT9/yfY4ww==", + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/@graphql-codegen/plugin-helpers/-/plugin-helpers-5.0.3.tgz", + "integrity": "sha512-yZ1rpULIWKBZqCDlvGIJRSyj1B2utkEdGmXZTBT/GVayP4hyRYlkd36AJV/LfEsVD8dnsKL5rLz2VTYmRNlJ5Q==", "dev": true, "requires": { "@graphql-tools/utils": "^10.0.0", @@ -18945,54 +19203,50 @@ "common-tags": "1.8.2", "import-from": "4.0.0", "lodash": "~4.17.0", - "tslib": "~2.5.0" - }, - "dependencies": { - "tslib": { - "version": "2.5.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.3.tgz", - "integrity": "sha512-mSxlJJwl3BMEQCUNnxXBU9jP4JBktcEGhURcPR6VQVlnP0FdDEsIaz0C35dXNGLyRfrATNofF0F5p2KPxQgB+w==", - "dev": true - } + "tslib": "~2.6.0" } }, "@graphql-codegen/schema-ast": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/@graphql-codegen/schema-ast/-/schema-ast-4.0.0.tgz", - "integrity": "sha512-WIzkJFa9Gz28FITAPILbt+7A8+yzOyd1NxgwFh7ie+EmO9a5zQK6UQ3U/BviirguXCYnn+AR4dXsoDrSrtRA1g==", + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@graphql-codegen/schema-ast/-/schema-ast-4.0.2.tgz", + "integrity": "sha512-5mVAOQQK3Oz7EtMl/l3vOQdc2aYClUzVDHHkMvZlunc+KlGgl81j8TLa+X7ANIllqU4fUEsQU3lJmk4hXP6K7Q==", "dev": true, "requires": { - "@graphql-codegen/plugin-helpers": "^5.0.0", + "@graphql-codegen/plugin-helpers": "^5.0.3", "@graphql-tools/utils": "^10.0.0", - "tslib": "~2.5.0" - }, - "dependencies": { - "tslib": { - "version": "2.5.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.3.tgz", - "integrity": "sha512-mSxlJJwl3BMEQCUNnxXBU9jP4JBktcEGhURcPR6VQVlnP0FdDEsIaz0C35dXNGLyRfrATNofF0F5p2KPxQgB+w==", - "dev": true - } + "tslib": "~2.6.0" } }, "@graphql-codegen/typescript": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/@graphql-codegen/typescript/-/typescript-4.0.1.tgz", - "integrity": "sha512-3YziQ21dCVdnHb+Us1uDb3pA6eG5Chjv0uTK+bt9dXeMlwYBU8MbtzvQTo4qvzWVC1AxSOKj0rgfNu1xCXqJyA==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/@graphql-codegen/typescript/-/typescript-4.0.4.tgz", + "integrity": "sha512-x79CKLfP9UQCX+/I78qxQlMs2Mmq3pF1lKafZo7lAno0f/fvJ+qWUduzdgjRNz+YL+5blGeWcC0pWEDxniO7hw==", "dev": true, "requires": { - "@graphql-codegen/plugin-helpers": "^5.0.0", - "@graphql-codegen/schema-ast": "^4.0.0", - "@graphql-codegen/visitor-plugin-common": "4.0.1", + "@graphql-codegen/plugin-helpers": "^5.0.3", + "@graphql-codegen/schema-ast": "^4.0.2", + "@graphql-codegen/visitor-plugin-common": "4.1.2", "auto-bind": "~4.0.0", - "tslib": "~2.5.0" + "tslib": "~2.6.0" }, "dependencies": { - "tslib": { - "version": "2.5.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.3.tgz", - "integrity": "sha512-mSxlJJwl3BMEQCUNnxXBU9jP4JBktcEGhURcPR6VQVlnP0FdDEsIaz0C35dXNGLyRfrATNofF0F5p2KPxQgB+w==", - "dev": true + "@graphql-codegen/visitor-plugin-common": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/@graphql-codegen/visitor-plugin-common/-/visitor-plugin-common-4.1.2.tgz", + "integrity": "sha512-yk7iEAL1kYZ2Gi/pvVjdsZhul5WsYEM4Zcgh2Ev15VicMdJmPHsMhNUsZWyVJV0CaQCYpNOFlGD/11Ea3pn4GA==", + "dev": true, + "requires": { + "@graphql-codegen/plugin-helpers": "^5.0.3", + "@graphql-tools/optimize": "^2.0.0", + "@graphql-tools/relay-operation-optimizer": "^7.0.0", + "@graphql-tools/utils": "^10.0.0", + "auto-bind": "~4.0.0", + "change-case-all": "1.0.15", + "dependency-graph": "^0.11.0", + "graphql-tag": "^2.11.0", + "parse-filepath": "^1.0.2", + "tslib": "~2.6.0" + } } } }, @@ -21607,9 +21861,9 @@ "integrity": "sha512-WZLiwShhwLRmeV6zH+GkbOFT6Z6VklCItrDioxUnv+u4Ll+8vKeFySoFyK/0ctcRpOmwAicELfmys1sDc/Rw+A==" }, "@types/semver": { - "version": "7.5.6", - "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.6.tgz", - "integrity": "sha512-dn1l8LaMea/IjDoHNd9J52uBbInB796CDffS6VdIxvqYCPSG0V0DzHp76GpaWnlhg88uYyPbXCDIowa86ybd5A==", + "version": "7.5.7", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.7.tgz", + "integrity": "sha512-/wdoPq1QqkSj9/QOeKkFquEuPzQbHTWAMPH/PaUMB+JuR31lXhlWXRZ52IpfDYVlDOUBvX09uBrPwxGT1hjNBg==", "dev": true }, "@types/send": { @@ -21658,9 +21912,9 @@ "dev": true }, "@types/validator": { - "version": "13.11.8", - "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.11.8.tgz", - "integrity": "sha512-c/hzNDBh7eRF+KbCf+OoZxKbnkpaK/cKp9iLQWqB7muXtM+MtL9SUUH8vCFcLn6dH1Qm05jiexK0ofWY7TfOhQ==", + "version": "13.11.9", + "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.11.9.tgz", + "integrity": "sha512-FCTsikRozryfayPuiI46QzH3fnrOoctTjvOYZkho9BTFLCOZ2rgZJHMOVgCOfttjPJcgOx52EpkY0CMfy87MIw==", "dev": true }, "@types/ws": { @@ -21685,16 +21939,16 @@ "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==" }, "@typescript-eslint/eslint-plugin": { - "version": "6.20.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.20.0.tgz", - "integrity": "sha512-fTwGQUnjhoYHeSF6m5pWNkzmDDdsKELYrOBxhjMrofPqCkoC2k3B2wvGHFxa1CTIqkEn88nlW1HVMztjo2K8Hg==", + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.21.0.tgz", + "integrity": "sha512-oy9+hTPCUFpngkEZUSzbf9MxI65wbKFoQYsgPdILTfbUldp5ovUuphZVe4i30emU9M/kP+T64Di0mxl7dSw3MA==", "dev": true, "requires": { "@eslint-community/regexpp": "^4.5.1", - "@typescript-eslint/scope-manager": "6.20.0", - "@typescript-eslint/type-utils": "6.20.0", - "@typescript-eslint/utils": "6.20.0", - "@typescript-eslint/visitor-keys": "6.20.0", + "@typescript-eslint/scope-manager": "6.21.0", + "@typescript-eslint/type-utils": "6.21.0", + "@typescript-eslint/utils": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", "debug": "^4.3.4", "graphemer": "^1.4.0", "ignore": "^5.2.4", @@ -21703,6 +21957,32 @@ "ts-api-utils": "^1.0.1" }, "dependencies": { + "@typescript-eslint/scope-manager": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.21.0.tgz", + "integrity": "sha512-OwLUIWZJry80O99zvqXVEioyniJMa+d2GrqpUTqi5/v5D5rOrppJVBPa0yKCblcigC0/aYAzxxqQ1B+DS2RYsg==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0" + } + }, + "@typescript-eslint/types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", + "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", + "dev": true + }, + "@typescript-eslint/visitor-keys": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", + "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.21.0", + "eslint-visitor-keys": "^3.4.1" + } + }, "lru-cache": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", @@ -21747,15 +22027,85 @@ } }, "@typescript-eslint/type-utils": { - "version": "6.20.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.20.0.tgz", - "integrity": "sha512-qnSobiJQb1F5JjN0YDRPHruQTrX7ICsmltXhkV536mp4idGAYrIyr47zF/JmkJtEcAVnIz4gUYJ7gOZa6SmN4g==", + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.21.0.tgz", + "integrity": "sha512-rZQI7wHfao8qMX3Rd3xqeYSMCL3SoiSQLBATSiVKARdFGCYSRvmViieZjqc58jKgs8Y8i9YvVVhRbHSTA4VBag==", "dev": true, "requires": { - "@typescript-eslint/typescript-estree": "6.20.0", - "@typescript-eslint/utils": "6.20.0", + "@typescript-eslint/typescript-estree": "6.21.0", + "@typescript-eslint/utils": "6.21.0", "debug": "^4.3.4", "ts-api-utils": "^1.0.1" + }, + "dependencies": { + "@typescript-eslint/types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", + "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", + "dev": true + }, + "@typescript-eslint/typescript-estree": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.21.0.tgz", + "integrity": "sha512-6npJTkZcO+y2/kr+z0hc4HwNfrrP4kNYh57ek7yCNlrBjWQ1Y0OS7jiZTkgumrvkX5HkEKXFZkkdFNkaW2wmUQ==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "minimatch": "9.0.3", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + } + }, + "@typescript-eslint/visitor-keys": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", + "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.21.0", + "eslint-visitor-keys": "^3.4.1" + } + }, + "brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0" + } + }, + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "minimatch": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", + "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "dev": true, + "requires": { + "brace-expansion": "^2.0.1" + } + }, + "semver": { + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } } }, "@typescript-eslint/types": { @@ -21819,20 +22169,71 @@ } }, "@typescript-eslint/utils": { - "version": "6.20.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.20.0.tgz", - "integrity": "sha512-/EKuw+kRu2vAqCoDwDCBtDRU6CTKbUmwwI7SH7AashZ+W+7o8eiyy6V2cdOqN49KsTcASWsC5QeghYuRDTyOOg==", + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.21.0.tgz", + "integrity": "sha512-NfWVaC8HP9T8cbKQxHcsJBY5YE1O33+jpMwN45qzWWaPDZgLIbo12toGMWnmhvCpd3sIxkpDw3Wv1B3dYrbDQQ==", "dev": true, "requires": { "@eslint-community/eslint-utils": "^4.4.0", "@types/json-schema": "^7.0.12", "@types/semver": "^7.5.0", - "@typescript-eslint/scope-manager": "6.20.0", - "@typescript-eslint/types": "6.20.0", - "@typescript-eslint/typescript-estree": "6.20.0", + "@typescript-eslint/scope-manager": "6.21.0", + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/typescript-estree": "6.21.0", "semver": "^7.5.4" }, "dependencies": { + "@typescript-eslint/scope-manager": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.21.0.tgz", + "integrity": "sha512-OwLUIWZJry80O99zvqXVEioyniJMa+d2GrqpUTqi5/v5D5rOrppJVBPa0yKCblcigC0/aYAzxxqQ1B+DS2RYsg==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0" + } + }, + "@typescript-eslint/types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", + "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", + "dev": true + }, + "@typescript-eslint/typescript-estree": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.21.0.tgz", + "integrity": "sha512-6npJTkZcO+y2/kr+z0hc4HwNfrrP4kNYh57ek7yCNlrBjWQ1Y0OS7jiZTkgumrvkX5HkEKXFZkkdFNkaW2wmUQ==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "minimatch": "9.0.3", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + } + }, + "@typescript-eslint/visitor-keys": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", + "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", + "dev": true, + "requires": { + "@typescript-eslint/types": "6.21.0", + "eslint-visitor-keys": "^3.4.1" + } + }, + "brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0" + } + }, "lru-cache": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", @@ -21842,10 +22243,19 @@ "yallist": "^4.0.0" } }, + "minimatch": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", + "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "dev": true, + "requires": { + "brace-expansion": "^2.0.1" + } + }, "semver": { - "version": "7.5.4", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", - "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", "dev": true, "requires": { "lru-cache": "^6.0.0" @@ -25980,9 +26390,9 @@ "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==" }, "lint-staged": { - "version": "15.2.1", - "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.2.1.tgz", - "integrity": "sha512-dhwAPnM85VdshybV9FWI/9ghTvMLoQLEXgVMx+ua2DN7mdfzd/tRfoU2yhMcBac0RHkofoxdnnJUokr8s4zKmQ==", + "version": "15.2.2", + "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-15.2.2.tgz", + "integrity": "sha512-TiTt93OPh1OZOsb5B7k96A/ATl2AjIZo+vnzFZ6oHK5FuTk63ByDtxGQpHm+kFETjEWqgkF95M8FRXKR/LEBcw==", "dev": true, "requires": { "chalk": "5.3.0", @@ -26879,9 +27289,9 @@ "integrity": "sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==" }, "nodemailer": { - "version": "6.9.8", - "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.9.8.tgz", - "integrity": "sha512-cfrYUk16e67Ks051i4CntM9kshRYei1/o/Gi8K1d+R34OIs21xdFnW7Pt7EucmVKA0LKtqUGNcjMZ7ehjl49mQ==" + "version": "6.9.9", + "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.9.9.tgz", + "integrity": "sha512-dexTll8zqQoVJEZPwQAKzxxtFn0qTnjdQTchoU6Re9BUUGBJiOy3YMn/0ShTW6J5M0dfQ1NeDeRTTl4oIWgQMA==" }, "normalize-path": { "version": "3.0.0", diff --git a/package.json b/package.json index 570b502bd9..eabaa960ee 100644 --- a/package.json +++ b/package.json @@ -83,7 +83,7 @@ "mongoose-paginate-v2": "^1.7.0", "morgan": "^1.10.0", "nanoid": "^5.0.5", - "nodemailer": "^6.9.8", + "nodemailer": "^6.9.9", "pm2": "^5.2.0", "redis": "^4.6.12", "shortid": "^2.2.16", @@ -98,7 +98,7 @@ }, "devDependencies": { "@graphql-codegen/cli": "^5.0.0", - "@graphql-codegen/typescript": "^4.0.1", + "@graphql-codegen/typescript": "^4.0.4", "@graphql-codegen/typescript-resolvers": "^4.0.1", "@graphql-eslint/eslint-plugin": "^3.20.1", "@types/bcryptjs": "^2.4.6", @@ -118,8 +118,8 @@ "@types/nodemailer": "^6.4.14", "@types/shortid": "^0.0.32", "@types/uuid": "^9.0.7", - "@types/validator": "^13.11.7", - "@typescript-eslint/eslint-plugin": "^6.20.0", + "@types/validator": "^13.11.9", + "@typescript-eslint/eslint-plugin": "^6.21.0", "@typescript-eslint/parser": "^6.20.0", "@vitest/coverage-c8": "^0.24.3", "concurrently": "^8.2.2", @@ -130,7 +130,7 @@ "get-graphql-schema": "^2.1.2", "graphql-markdown": "^7.0.0", "husky": "^9.0.6", - "lint-staged": "^15.2.1", + "lint-staged": "^15.2.2", "prettier": "^2.7.1", "rimraf": "^5.0.5", "tsx": "^4.7.0", diff --git a/sample_data/appUserProfiles.json b/sample_data/appUserProfiles.json new file mode 100644 index 0000000000..79a0d00453 --- /dev/null +++ b/sample_data/appUserProfiles.json @@ -0,0 +1,107 @@ +[ + { + "_id": "61378abd85008f171cf2990d", + "userId": "64378abd85008f171cf2990d", + "adminFor": ["6437904485008f171cf29924"], + "appLanguageCode": "en", + "createdEvents": [], + "createdOrganizations": ["6437904485008f171cf29924"], + "eventAdmin": [], + "pluginCreationAllowed": true, + "token": null, + "tokenVersion": 0, + "isSuperAdmin": true, + "createdAt": "2023-04-13T04:53:17.742Z", + "__v": 0 + }, + { + "_id": "65978abd85008f171cf2990d", + "userId": "65378abd85008f171cf2990d", + "adminFor": ["6537904485008f171cf29924"], + "appLanguageCode": "en", + "createdEvents": [], + "createdOrganizations": [], + "eventAdmin": [], + "pluginCreationAllowed": true, + "token": null, + "tokenVersion": 0, + "isSuperAdmin": false, + "createdAt": "2023-04-13T04:53:17.742Z", + "__v": 0 + }, + { + "_id": "66478abd85008f171cf2990d", + "userId": "66378abd85008f171cf2990d", + "adminFor": ["6637904485008f171cf29924"], + "appLanguageCode": "en", + "createdEvents": [], + "createdOrganizations": [], + "eventAdmin": [], + "pluginCreationAllowed": true, + "token": null, + "tokenVersion": 0, + "isSuperAdmin": false, + "createdAt": "2023-04-13T04:53:17.742Z", + "__v": 0 + }, + { + "_id": "67379abd85008f171cf2990d", + "userId": "67378abd85008f171cf2990d", + "adminFor": ["6737904485008f171cf29924"], + "appLanguageCode": "en", + "createdEvents": [], + "createdOrganizations": [], + "eventAdmin": [], + "pluginCreationAllowed": true, + "token": null, + "tokenVersion": 0, + "isSuperAdmin": false, + "createdAt": "2023-04-13T04:53:17.742Z", + "__v": 0 + }, + { + "_id": "67078abd85008f171cf2991d", + "userId": "67378abd85008f171cf2991d", + "adminFor": [], + "appLanguageCode": "en", + "createdEvents": [], + "createdOrganizations": [], + "eventAdmin": [], + "pluginCreationAllowed": true, + "token": null, + "tokenVersion": 0, + "isSuperAdmin": false, + "createdAt": "2023-04-13T04:53:17.742Z", + "__v": 0 + }, + { + "_id": "62378abd85008f171cf2992d", + "userId": "67378abd85008f171cf2992d", + "adminFor": [], + "appLanguageCode": "en", + "createdEvents": [], + "createdOrganizations": [], + "eventAdmin": [], + "pluginCreationAllowed": true, + "token": null, + "tokenVersion": 0, + "isSuperAdmin": false, + "createdAt": "2023-04-13T04:53:17.742Z", + "__v": 0 + }, + { + "_id": "69378abd85008f171cf2993d", + "userId": "67378abd85008f171cf2993d", + "adminFor": [], + "appLanguageCode": "en", + "createdEvents": [], + "createdOrganizations": [], + "eventAdmin": [], + "pluginCreationAllowed": true, + "token": null, + "tokenVersion": 0, + "isSuperAdmin": false, + "createdAt": "2023-04-13T04:53:17.742Z", + "__v": 0 + } +] diff --git a/sample_data/users.json b/sample_data/users.json index d3fa89a564..d92fdef86c 100644 --- a/sample_data/users.json +++ b/sample_data/users.json @@ -1,15 +1,9 @@ [ { "_id": "64378abd85008f171cf2990d", - "tokenVersion": 0, - "appLanguageCode": "en", - "createdOrganizations": ["6437904485008f171cf29924"], - "createdEvents": [], - "userType": "SUPERADMIN", + "appUserProfileId": "61378abd85008f171cf2990d", "joinedOrganizations": ["6437904485008f171cf29924"], "registeredEvents": [], - "eventAdmin": [], - "adminFor": ["6437904485008f171cf29924"], "membershipRequests": [], "organizationsBlockedBy": [], "status": "ACTIVE", @@ -25,15 +19,9 @@ }, { "_id": "65378abd85008f171cf2990d", - "tokenVersion": 0, - "appLanguageCode": "en", - "createdOrganizations": [], - "createdEvents": [], - "userType": "ADMIN", + "appUserProfileId": "65978abd85008f171cf2990d", "joinedOrganizations": ["6537904485008f171cf29924"], "registeredEvents": [], - "eventAdmin": [], - "adminFor": ["6537904485008f171cf29924"], "membershipRequests": [], "organizationsBlockedBy": [], "status": "ACTIVE", @@ -49,15 +37,9 @@ }, { "_id": "66378abd85008f171cf2990d", - "tokenVersion": 0, - "appLanguageCode": "en", - "createdOrganizations": [], - "createdEvents": [], - "userType": "ADMIN", + "appUserProfileId": "66478abd85008f171cf2990d", "joinedOrganizations": ["6637904485008f171cf29924"], "registeredEvents": [], - "eventAdmin": [], - "adminFor": ["6637904485008f171cf29924"], "membershipRequests": [], "organizationsBlockedBy": [], "status": "ACTIVE", @@ -73,15 +55,9 @@ }, { "_id": "67378abd85008f171cf2990d", - "tokenVersion": 0, - "appLanguageCode": "en", - "createdOrganizations": [], - "createdEvents": [], - "userType": "ADMIN", + "appUserProfileId": "67379abd85008f171cf2990d", "joinedOrganizations": ["6737904485008f171cf29924"], "registeredEvents": [], - "eventAdmin": [], - "adminFor": ["6737904485008f171cf29924"], "membershipRequests": [], "organizationsBlockedBy": [], "status": "ACTIVE", @@ -97,15 +73,9 @@ }, { "_id": "67378abd85008f171cf2991d", - "tokenVersion": 0, - "appLanguageCode": "en", - "createdOrganizations": [], - "createdEvents": [], - "userType": "USER", + "appUserProfileId": "67078abd85008f171cf2991d", "joinedOrganizations": ["6537904485008f171cf29924"], "registeredEvents": [], - "eventAdmin": [], - "adminFor": [], "membershipRequests": [], "organizationsBlockedBy": [], "status": "ACTIVE", @@ -121,15 +91,9 @@ }, { "_id": "67378abd85008f171cf2992d", - "tokenVersion": 0, - "appLanguageCode": "en", - "createdOrganizations": [], - "createdEvents": [], - "userType": "USER", + "appUserProfileId": "62378abd85008f171cf2992d", "joinedOrganizations": ["6537904485008f171cf29924"], "registeredEvents": [], - "eventAdmin": [], - "adminFor": [], "membershipRequests": [], "organizationsBlockedBy": [], "status": "ACTIVE", @@ -145,15 +109,9 @@ }, { "_id": "67378abd85008f171cf2993d", - "tokenVersion": 0, - "appLanguageCode": "en", - "createdOrganizations": [], - "createdEvents": [], - "userType": "USER", + "appUserProfileId": "69378abd85008f171cf2993d", "joinedOrganizations": ["6537904485008f171cf29924"], "registeredEvents": [], - "eventAdmin": [], - "adminFor": [], "membershipRequests": [], "organizationsBlockedBy": [], "status": "ACTIVE", diff --git a/schema.graphql b/schema.graphql index 19d7ad4b4d..c5023def45 100644 --- a/schema.graphql +++ b/schema.graphql @@ -80,8 +80,20 @@ type AggregateUser { scalar Any +type AppUserProfile { + _id: ID! + adminFor: [Organization] + createdEvents: [Event] + createdOrganizations: [Organization] + eventAdmin: [Event] + isSuperAdmin: Boolean! + pluginCreationAllowed: Boolean! + userId: User! +} + type AuthData { accessToken: String! + appUserProfile: AppUserProfile! refreshToken: String! user: User! } @@ -525,13 +537,13 @@ type Mutation { adminRemoveEvent(eventId: ID!): Event! adminRemoveGroup(groupId: ID!): GroupChat! assignUserTag(input: ToggleUserTagAssignInput!): User - blockPluginCreationBySuperadmin(blockUser: Boolean!, userId: ID!): User! + blockPluginCreationBySuperadmin(blockUser: Boolean!, userId: ID!): AppUserProfile! blockUser(organizationId: ID!, userId: ID!): User! cancelMembershipRequest(membershipRequestId: ID!): MembershipRequest! checkIn(data: CheckInInput!): CheckIn! createActionItem(actionItemCategoryId: ID!, data: CreateActionItemInput!): ActionItem! createActionItemCategory(name: String!, organizationId: ID!): ActionItemCategory! - createAdmin(data: UserAndOrganizationInput!): User! + createAdmin(data: UserAndOrganizationInput!): AppUserProfile! createAdvertisement(endDate: Date!, link: String!, name: String!, orgId: ID!, startDate: Date!, type: String!): Advertisement! createComment(data: CommentInput!, postId: ID!): Comment createDirectChat(data: createChatInput!): DirectChat! @@ -562,7 +574,7 @@ type Mutation { rejectAdmin(id: ID!): Boolean! rejectMembershipRequest(membershipRequestId: ID!): MembershipRequest! removeActionItem(id: ID!): ActionItem! - removeAdmin(data: UserAndOrganizationInput!): User! + removeAdmin(data: UserAndOrganizationInput!): AppUserProfile! removeAdvertisement(id: ID!): Advertisement removeComment(id: ID!): Comment removeDirectChat(chatId: ID!, organizationId: ID!): DirectChat! @@ -570,7 +582,7 @@ type Mutation { removeEventAttendee(data: EventAttendeeInput!): User! removeGroupChat(chatId: ID!): GroupChat! removeMember(data: UserAndOrganizationInput!): Organization! - removeOrganization(id: ID!): User! + removeOrganization(id: ID!): UserData! removeOrganizationCustomField(customFieldId: ID!, organizationId: ID!): OrganizationCustomField! removeOrganizationImage(organizationId: String!): Organization! removePost(id: ID!): Post @@ -601,7 +613,7 @@ type Mutation { updateOrganization(data: UpdateOrganizationInput, file: String, id: ID!): Organization! updatePluginStatus(id: ID!, orgId: ID!): Plugin! updatePost(data: PostUpdateInput, id: ID!): Post! - updateUserPassword(data: UpdateUserPasswordInput!): User! + updateUserPassword(data: UpdateUserPasswordInput!): UserData! updateUserProfile(data: UpdateUserInput, file: String): User! updateUserRoleInOrganization(organizationId: ID!, role: String!, userId: ID!): Organization! updateUserTag(input: UpdateUserTagInput!): UserTag @@ -880,9 +892,9 @@ type Query { postsByOrganizationConnection(first: Int, id: ID!, orderBy: PostOrderByInput, skip: Int, where: PostWhereInput): PostConnection registeredEventsByUser(id: ID, orderBy: EventOrderByInput): [Event] registrantsByEvent(id: ID!): [User] - user(id: ID!): User! + user(id: ID!): UserData! userLanguage(userId: ID!): String - users(adminApproved: Boolean, first: Int, orderBy: UserOrderByInput, skip: Int, userType: String, where: UserWhereInput): [User] + users(adminApproved: Boolean, first: Int, orderBy: UserOrderByInput, skip: Int, where: UserWhereInput): [User] usersConnection(first: Int, orderBy: UserOrderByInput, skip: Int, where: UserWhereInput): [User]! } @@ -1027,16 +1039,13 @@ type User { _id: ID! address: Address adminApproved: Boolean - adminFor: [Organization] appLanguageCode: String! + appUserProfileId: AppUserProfile birthDate: Date createdAt: DateTime! - createdEvents: [Event] - createdOrganizations: [Organization] educationGrade: EducationGrade email: EmailAddress! employmentStatus: EmploymentStatus - eventAdmin: [Event] firstName: String! gender: Gender image: String @@ -1049,9 +1058,7 @@ type User { pluginCreationAllowed: Boolean! registeredEvents: [Event] tagsAssignedWith(after: String, before: String, first: PositiveInt, last: PositiveInt, organizationId: ID): UserTagsConnection - tokenVersion: Int! updatedAt: DateTime! - userType: UserType! } input UserAndOrganizationInput { @@ -1072,6 +1079,11 @@ type UserCustomData { values: JSON! } +type UserData { + appUserProfile: AppUserProfile! + user: User! +} + type UserEdge { cursor: String! node: User! @@ -1095,8 +1107,6 @@ input UserInput { } enum UserOrderByInput { - appLanguageCode_ASC - appLanguageCode_DESC email_ASC email_DESC firstName_ASC @@ -1157,13 +1167,6 @@ enum UserType { } input UserWhereInput { - admin_for: ID - appLanguageCode: String - appLanguageCode_contains: String - appLanguageCode_in: [String!] - appLanguageCode_not: String - appLanguageCode_not_in: [String!] - appLanguageCode_starts_with: String email: EmailAddress email_contains: EmailAddress email_in: [EmailAddress!] diff --git a/src/app.ts b/src/app.ts index e26d89fda3..6e7891963f 100644 --- a/src/app.ts +++ b/src/app.ts @@ -9,7 +9,7 @@ import requestLogger from "morgan"; import path from "path"; import { appConfig } from "./config"; import { requestContext, requestTracing, stream } from "./libraries"; -//@ts-ignore + import graphqlUploadExpress from "graphql-upload/graphqlUploadExpress.mjs"; const app = express(); @@ -21,6 +21,7 @@ const apiLimiter = rateLimit({ message: "Too many requests from this IP, please try again after 15 minutes", }); +// eslint-disable-next-line @typescript-eslint/no-unused-vars const corsOptions: cors.CorsOptions = { origin: (origin, next) => { if (process.env.NODE_ENV === "development") { diff --git a/src/directives/directiveTransformer/authDirectiveTransformer.ts b/src/directives/directiveTransformer/authDirectiveTransformer.ts index a0bcb1cb2d..871dde918f 100644 --- a/src/directives/directiveTransformer/authDirectiveTransformer.ts +++ b/src/directives/directiveTransformer/authDirectiveTransformer.ts @@ -1,13 +1,14 @@ import { MapperKind, getDirective, mapSchema } from "@graphql-tools/utils"; -import type { GraphQLFieldConfig } from "graphql"; +import { defaultFieldResolver } from "graphql"; +import type { GraphQLSchema } from "graphql/type/schema"; import { errors, requestContext } from "../../libraries"; -//@ts-ignore -function authDirectiveTransformer(schema, directiveName): any { +function authDirectiveTransformer( + schema: GraphQLSchema, + directiveName: string +): GraphQLSchema { return mapSchema(schema, { - [MapperKind.OBJECT_FIELD]: ( - fieldConfig: GraphQLFieldConfig - ): any => { + [MapperKind.OBJECT_FIELD]: (fieldConfig) => { // Check whether this field has the specified directive const authDirective = getDirective( schema, @@ -15,7 +16,6 @@ function authDirectiveTransformer(schema, directiveName): any { directiveName )?.[0]; if (authDirective) { - //@ts-ignore const { resolve = defaultFieldResolver } = fieldConfig; fieldConfig.resolve = (root, args, context, info): string => { @@ -25,7 +25,7 @@ function authDirectiveTransformer(schema, directiveName): any { "user.notAuthenticated --auth directive", "userAuthentication" ); - return resolve(root, args, context, info); + return resolve(root, args, context, info) as string; }; return fieldConfig; } diff --git a/src/directives/directiveTransformer/roleDirectiveTransformer.ts b/src/directives/directiveTransformer/roleDirectiveTransformer.ts index ee424c1fef..0f00dbeb8d 100644 --- a/src/directives/directiveTransformer/roleDirectiveTransformer.ts +++ b/src/directives/directiveTransformer/roleDirectiveTransformer.ts @@ -1,18 +1,16 @@ -import type { GraphQLFieldConfig } from "graphql"; -import { - USER_NOT_AUTHORIZED_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../constants"; +import { MapperKind, getDirective, mapSchema } from "@graphql-tools/utils"; +import type { GraphQLSchema } from "graphql"; +import { defaultFieldResolver } from "graphql"; +import { USER_NOT_FOUND_ERROR } from "../../constants"; import { errors } from "../../libraries"; import { User } from "../../models"; -import { MapperKind, getDirective, mapSchema } from "@graphql-tools/utils"; -//@ts-ignore -function roleDirectiveTransformer(schema, directiveName): any { +function roleDirectiveTransformer( + schema: GraphQLSchema, + directiveName: string +): GraphQLSchema { return mapSchema(schema, { - [MapperKind.OBJECT_FIELD]: ( - fieldConfig: GraphQLFieldConfig - ): any => { + [MapperKind.OBJECT_FIELD]: (fieldConfig) => { // Check whether this field has the specified directive const roleDirective = getDirective( schema, @@ -21,11 +19,9 @@ function roleDirectiveTransformer(schema, directiveName): any { )?.[0]; if (roleDirective) { - //@ts-ignore - const { resolve = defaultFieldResolver } = fieldConfig; - const { requires } = roleDirective; + // const { requires } = roleDirective; fieldConfig.resolve = async ( root, @@ -45,17 +41,17 @@ function roleDirectiveTransformer(schema, directiveName): any { ); } - if (currentUser.userType !== requires) { - throw new errors.UnauthenticatedError( - USER_NOT_AUTHORIZED_ERROR.MESSAGE, - USER_NOT_AUTHORIZED_ERROR.CODE, - USER_NOT_AUTHORIZED_ERROR.PARAM - ); - } + // if (currentUser.userType !== requires) { + // throw new errors.UnauthenticatedError( + // USER_NOT_AUTHORIZED_ERROR.MESSAGE, + // USER_NOT_AUTHORIZED_ERROR.CODE, + // USER_NOT_AUTHORIZED_ERROR.PARAM + // ); + // } context.user = currentUser; - return resolve(root, args, context, info); + return resolve(root, args, context, info) as string; }; return fieldConfig; diff --git a/src/models/Advertisement.ts b/src/models/Advertisement.ts index 2aacecf2e8..ef1a48188b 100644 --- a/src/models/Advertisement.ts +++ b/src/models/Advertisement.ts @@ -2,7 +2,7 @@ import type { Types, Model, PopulatedDoc } from "mongoose"; import { Schema, model, models } from "mongoose"; import type { InterfaceUser } from "./User"; /** - * This is an interface that represents a database(MongoDB) document for Advertisement. + * This is an interface, that represents database - (MongoDB) document for Advertisement. */ export interface InterfaceAdvertisement { _id: Types.ObjectId; diff --git a/src/models/AppUserProfile.ts b/src/models/AppUserProfile.ts new file mode 100644 index 0000000000..eff51c6b17 --- /dev/null +++ b/src/models/AppUserProfile.ts @@ -0,0 +1,103 @@ +import type { PaginateModel, PopulatedDoc, Types } from "mongoose"; +import { Schema, model, models } from "mongoose"; +import mongoosePaginate from "mongoose-paginate-v2"; +import type { InterfaceEvent } from "./Event"; +import type { InterfaceOrganization } from "./Organization"; +import type { InterfaceUser } from "./User"; + +export interface InterfaceAppUserProfile { + _id: Types.ObjectId; + userId: PopulatedDoc; + adminFor: PopulatedDoc[]; + appLanguageCode: string; + createdEvents: PopulatedDoc[]; + createdOrganizations: PopulatedDoc[]; + eventAdmin: PopulatedDoc[]; + pluginCreationAllowed: boolean; + token: string | undefined; + tokenVersion: number; + isSuperAdmin: boolean; +} +/** + * This describes the schema for a `AppUserProfile` that corresponds to `InterfaceAppUserProfile` document. + * @param user - User id of the AppUserProfile + * @param adminFor - Collection of organization where appuser is admin, each object refer to `Organization` model. + * @param appLanguageCode - AppUser's app language code. + * @param createdEvents - Collection of all events created by the user, each object refer to `Event` model. + * @param createdOrganizations - Collection of all organization created by the user, each object refer to `Organization` model. + * @param eventAdmin - Collection of the event admins, each object refer to `Event` model. + * @param pluginCreationAllowed - Wheather user is allowed to create plugins. + * @param tokenVersion - Token version. + * @param isSuperAdmin - Wheather user is super admin. + * @param token - Access token. + * */ + +const appUserSchema = new Schema( + { + userId: { + type: Schema.Types.ObjectId, + ref: "User", + required: true, + }, + adminFor: [ + { + type: Schema.Types.ObjectId, + ref: "Organization", + }, + ], + appLanguageCode: { + type: String, + required: true, + default: "en", + }, + createdOrganizations: [ + { + type: Schema.Types.ObjectId, + ref: "Organization", + }, + ], + createdEvents: [ + { + type: Schema.Types.ObjectId, + ref: "Event", + }, + ], + eventAdmin: [ + { + type: Schema.Types.ObjectId, + ref: "Event", + }, + ], + pluginCreationAllowed: { + type: Boolean, + required: true, + default: true, + }, + tokenVersion: { + type: Number, + required: true, + default: 0, + }, + token: { + type: String, + required: false, + }, + isSuperAdmin: { + type: Boolean, + required: true, + default: false, + }, + }, + { + timestamps: true, + } +); +appUserSchema.plugin(mongoosePaginate); +const appUserProfileModel = (): PaginateModel => + model>( + "AppUserProfile", + appUserSchema + ); + +export const AppUserProfile = (models.AppUserProfile || + appUserProfileModel()) as ReturnType; diff --git a/src/models/MembershipRequest.ts b/src/models/MembershipRequest.ts index 6118a176cd..671fe013ec 100644 --- a/src/models/MembershipRequest.ts +++ b/src/models/MembershipRequest.ts @@ -1,4 +1,4 @@ -import type { PopulatedDoc, Types, Document, Model } from "mongoose"; +import type { Document, Model, PopulatedDoc, Types } from "mongoose"; import { Schema, model, models } from "mongoose"; import type { InterfaceOrganization } from "./Organization"; import type { InterfaceUser } from "./User"; diff --git a/src/models/SampleData.ts b/src/models/SampleData.ts index f6a4bf2b9f..583bd9ea57 100644 --- a/src/models/SampleData.ts +++ b/src/models/SampleData.ts @@ -3,7 +3,13 @@ import { Schema, model, models } from "mongoose"; export interface InterfaceSampleData { documentId: string; - collectionName: "Organization" | "Post" | "Event" | "User" | "Plugin"; + collectionName: + | "Organization" + | "Post" + | "Event" + | "User" + | "Plugin" + | "AppUserProfile"; } const sampleDataSchema = new Schema({ @@ -14,7 +20,7 @@ const sampleDataSchema = new Schema({ collectionName: { type: String, required: true, - enum: ["Organization", "Post", "Event", "User", "Plugin"], + enum: ["Organization", "Post", "Event", "User", "AppUserProfile", "Plugin"], }, }); diff --git a/src/models/User.ts b/src/models/User.ts index befe7ee2f3..2a4000f8ed 100644 --- a/src/models/User.ts +++ b/src/models/User.ts @@ -1,17 +1,19 @@ -import type { PopulatedDoc, PaginateModel, Types, Document } from "mongoose"; +import type { Document, PaginateModel, PopulatedDoc, Types } from "mongoose"; import { Schema, model, models } from "mongoose"; import mongoosePaginate from "mongoose-paginate-v2"; import validator from "validator"; +import { createLoggingMiddleware } from "../libraries/dbLogger"; +import type { InterfaceAppUserProfile } from "./AppUserProfile"; import type { InterfaceEvent } from "./Event"; import type { InterfaceMembershipRequest } from "./MembershipRequest"; import type { InterfaceOrganization } from "./Organization"; -import { createLoggingMiddleware } from "../libraries/dbLogger"; /** * This is an interface that represents a database(MongoDB) document for User. */ export interface InterfaceUser { _id: Types.ObjectId; + appUserProfileId: PopulatedDoc; address: { city: string; countryCode: string; @@ -23,16 +25,14 @@ export interface InterfaceUser { state: string; }; adminApproved: boolean; - adminFor: PopulatedDoc[]; - appLanguageCode: string; + birthDate: Date; createdAt: Date; - createdEvents: PopulatedDoc[]; - createdOrganizations: PopulatedDoc[]; + educationGrade: string; email: string; employmentStatus: string; - eventAdmin: PopulatedDoc[]; + firstName: string; gender: string; image: string | undefined | null; @@ -41,34 +41,32 @@ export interface InterfaceUser { maritalStatus: string; membershipRequests: PopulatedDoc[]; organizationsBlockedBy: PopulatedDoc[]; - password: string; + password?: string; phone: { home: string; mobile: string; work: string; }; - pluginCreationAllowed: boolean; + registeredEvents: PopulatedDoc[]; status: string; - token: string | undefined; - tokenVersion: number; + updatedAt: Date; userType: string; } /** * This describes the schema for a `User` that corresponds to `InterfaceUser` document. + * @param appUserProfileId - AppUserProfile id of the User * @param address - User address * @param adminApproved - Wheather user is admin approved. - * @param adminFor - Collection of organization where user is admin, each object refer to `Organization` model. - * @param appLanguageCode - User's app language code. + * @param birthDate - User Date of birth * @param createdAt - Time stamp of data creation. - * @param createdEvents - Collection of all events created by the user, each object refer to `Event` model. - * @param createdOrganizations - Collection of all organization created by the user, each object refer to `Organization` model. + * @param educationGrade - User highest education degree * @param email - User email id. * @param employmentStatus - User employment status - * @param eventAdmin - Collection of the event admins, each object refer to `Event` model. + * @param firstName - User First Name. * @param gender - User gender * @param image - User Image URL. @@ -79,16 +77,20 @@ export interface InterfaceUser { * @param organizationsBlockedBy - Collections of organizations where user is blocked, each object refer to `Organization` model. * @param password - User hashed password. * @param phone - User contact numbers, for mobile, home and work - * @param pluginCreationAllowed - Wheather user is allowed to create plugins. + * @param registeredEvents - Collection of user registered Events, each object refer to `Event` model. * @param status - Status - * @param token - Access token. - * @param tokenVersion - Token version. + * + * @param updatedAt - Timestamp of data updation - * @param userType - User type. + */ const userSchema = new Schema( { + appUserProfileId: { + type: Schema.Types.ObjectId, + ref: "AppUserProfile", + }, address: { city: { type: String, @@ -119,32 +121,11 @@ const userSchema = new Schema( type: Boolean, default: false, }, - adminFor: [ - { - type: Schema.Types.ObjectId, - ref: "Organization", - }, - ], - appLanguageCode: { - type: String, - required: true, - default: "en", - }, + birthDate: { type: Date, }, - createdOrganizations: [ - { - type: Schema.Types.ObjectId, - ref: "Organization", - }, - ], - createdEvents: [ - { - type: Schema.Types.ObjectId, - ref: "Event", - }, - ], + educationGrade: { type: String, enum: [ @@ -176,12 +157,7 @@ const userSchema = new Schema( type: String, enum: ["FULL_TIME", "PART_TIME", "UNEMPLOYED", null], }, - eventAdmin: [ - { - type: Schema.Types.ObjectId, - ref: "Event", - }, - ], + firstName: { type: String, required: true, @@ -242,11 +218,7 @@ const userSchema = new Schema( type: String, }, }, - pluginCreationAllowed: { - type: Boolean, - required: true, - default: true, - }, + registeredEvents: [ { type: Schema.Types.ObjectId, @@ -259,21 +231,6 @@ const userSchema = new Schema( enum: ["ACTIVE", "BLOCKED", "DELETED"], default: "ACTIVE", }, - token: { - type: String, - required: false, - }, - tokenVersion: { - type: Number, - required: true, - default: 0, - }, - userType: { - type: String, - required: true, - enum: ["USER", "ADMIN", "SUPERADMIN", "NON_USER"], - default: "USER", - }, }, { timestamps: true, diff --git a/src/models/index.ts b/src/models/index.ts index cd287b1ec9..67425f859e 100644 --- a/src/models/index.ts +++ b/src/models/index.ts @@ -1,8 +1,8 @@ export * from "./ActionItem"; -export * from "./Advertisement"; export * from "./ActionItemCategory"; +export * from "./Advertisement"; +export * from "./AppUserProfile"; export * from "./CheckIn"; -export * from "./MessageChat"; export * from "./Comment"; export * from "./DirectChat"; export * from "./DirectChatMessage"; @@ -18,6 +18,7 @@ export * from "./ImageHash"; export * from "./Language"; export * from "./MembershipRequest"; export * from "./Message"; +export * from "./MessageChat"; export * from "./Organization"; export * from "./OrganizationCustomField"; export * from "./OrganizationTagUser"; @@ -25,5 +26,5 @@ export * from "./Plugin"; export * from "./PluginField"; export * from "./Post"; export * from "./SampleData"; -export * from "./User"; export * from "./TagUser"; +export * from "./User"; diff --git a/src/resolvers/Mutation/acceptAdmin.ts b/src/resolvers/Mutation/acceptAdmin.ts index 109a1476f3..5f942b90da 100644 --- a/src/resolvers/Mutation/acceptAdmin.ts +++ b/src/resolvers/Mutation/acceptAdmin.ts @@ -1,7 +1,10 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { USER_NOT_FOUND_ERROR } from "../../constants"; -import { User } from "../../models"; +import { + USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, +} from "../../constants"; import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { superAdminCheck } from "../../utilities/superAdminCheck"; /** * This function accepts the admin request sent by a user. @@ -10,7 +13,8 @@ import { superAdminCheck } from "../../utilities/superAdminCheck"; * @param context - context of entire application * @remarks THe following checks are done: * 1. Whether the user exists - * 2. Whether the user accepting the admin request is a superadmin or not. + * 2. Whether the user has appProfile or not (if not, then the user is not a superadmin). + * 3. Whether the user accepting the admin request is a superadmin or not. */ export const acceptAdmin: MutationResolvers["acceptAdmin"] = async ( _parent, @@ -28,8 +32,20 @@ export const acceptAdmin: MutationResolvers["acceptAdmin"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + //if user does not have appProfile then he is NON_USER + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE), + USER_NOT_AUTHORIZED_SUPERADMIN.CODE, + USER_NOT_AUTHORIZED_SUPERADMIN.PARAM + ); + } - superAdminCheck(currentUser); + superAdminCheck(currentUserAppProfile); const userExists = await User.exists({ _id: args.id, diff --git a/src/resolvers/Mutation/addEventAttendee.ts b/src/resolvers/Mutation/addEventAttendee.ts index 955eb3da69..106fd4a043 100644 --- a/src/resolvers/Mutation/addEventAttendee.ts +++ b/src/resolvers/Mutation/addEventAttendee.ts @@ -1,16 +1,16 @@ +import { Types } from "mongoose"; import { EVENT_NOT_FOUND_ERROR, + USER_ALREADY_REGISTERED_FOR_EVENT, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, - USER_ALREADY_REGISTERED_FOR_EVENT, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import type { InterfaceEvent } from "../../models"; -import { User, Event, EventAttendee } from "../../models"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { AppUserProfile, Event, EventAttendee, User } from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; -import { Types } from "mongoose"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const addEventAttendee: MutationResolvers["addEventAttendee"] = async ( _parent, @@ -28,6 +28,16 @@ export const addEventAttendee: MutationResolvers["addEventAttendee"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let event: InterfaceEvent | null; @@ -58,7 +68,7 @@ export const addEventAttendee: MutationResolvers["addEventAttendee"] = async ( admin === context.userID || Types.ObjectId(admin).equals(context.userId) ); - if (!isUserEventAdmin && currentUser.userType !== "SUPERADMIN") { + if (!isUserEventAdmin && !currentUserAppProfile.isSuperAdmin) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, diff --git a/src/resolvers/Mutation/addOrganizationCustomField.ts b/src/resolvers/Mutation/addOrganizationCustomField.ts index af5eb832aa..716e326838 100644 --- a/src/resolvers/Mutation/addOrganizationCustomField.ts +++ b/src/resolvers/Mutation/addOrganizationCustomField.ts @@ -1,3 +1,4 @@ +import { Types } from "mongoose"; import { CUSTOM_FIELD_NAME_MISSING, CUSTOM_FIELD_TYPE_MISSING, @@ -6,7 +7,12 @@ import { USER_NOT_FOUND_ERROR, } from "../../constants"; import { errors, requestContext } from "../../libraries"; -import { OrganizationCustomField, Organization, User } from "../../models"; +import { + AppUserProfile, + Organization, + OrganizationCustomField, + User, +} from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** @@ -16,9 +22,10 @@ import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; * @param context - context of entire application * @remarks The following checks are done: * 1. If the user exists - * 2. If the organization exists. - * 3. If the user is an admin for the organization. - * 4. If the required name and value was provided for the new custom field + * 2. If the user has appProfile + * 3. If the organization exists. + * 4. If the user is an admin for the organization. + * 5. If the required name and value was provided for the new custom field * @returns Newly Added Custom Field. */ @@ -35,6 +42,17 @@ export const addOrganizationCustomField: MutationResolvers["addOrganizationCusto USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const organization = await Organization.findOne({ _id: args.organizationId, @@ -48,12 +66,13 @@ export const addOrganizationCustomField: MutationResolvers["addOrganizationCusto ); } - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(organization._id) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (orgId) => + orgId && Types.ObjectId(orgId.toString()).equals(organization._id) ); if ( - !(currentUserIsOrganizationAdmin || currentUser.userType === "SUPERADMIN") + !(currentUserIsOrganizationAdmin || currentUserAppProfile.isSuperAdmin) ) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), diff --git a/src/resolvers/Mutation/addOrganizationImage.ts b/src/resolvers/Mutation/addOrganizationImage.ts index b466a82431..578fe6b4dc 100644 --- a/src/resolvers/Mutation/addOrganizationImage.ts +++ b/src/resolvers/Mutation/addOrganizationImage.ts @@ -1,12 +1,14 @@ import "dotenv/config"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { ORGANIZATION_NOT_FOUND_ERROR } from "../../constants"; import { errors, requestContext } from "../../libraries"; -import { adminCheck } from "../../utilities"; +import type { InterfaceOrganization } from "../../models"; import { Organization } from "../../models"; -import { ORGANIZATION_NOT_FOUND_ERROR } from "../../constants"; -import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { adminCheck } from "../../utilities"; +import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; + /** * This function adds Organization Image. * @param _parent - parent of current request @@ -70,5 +72,5 @@ export const addOrganizationImage: MutationResolvers["addOrganizationImage"] = await cacheOrganizations([updatedOrganization]); } - return updatedOrganization!; + return updatedOrganization as InterfaceOrganization; }; diff --git a/src/resolvers/Mutation/addUserToGroupChat.ts b/src/resolvers/Mutation/addUserToGroupChat.ts index ad8b49981f..47c20ce191 100644 --- a/src/resolvers/Mutation/addUserToGroupChat.ts +++ b/src/resolvers/Mutation/addUserToGroupChat.ts @@ -1,16 +1,16 @@ import "dotenv/config"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { adminCheck } from "../../utilities"; -import { User, GroupChat, Organization } from "../../models"; import { CHAT_NOT_FOUND_ERROR, - USER_ALREADY_MEMBER_ERROR, ORGANIZATION_NOT_FOUND_ERROR, + USER_ALREADY_MEMBER_ERROR, USER_NOT_FOUND_ERROR, } from "../../constants"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import { errors, requestContext } from "../../libraries"; +import { GroupChat, Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { adminCheck } from "../../utilities"; /** * This function adds user to group chat. * @param _parent - parent of current request @@ -50,8 +50,9 @@ export const addUserToGroupChat: MutationResolvers["addUserToGroupChat"] = organization = await Organization.findOne({ _id: groupChat.organization, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) { + await cacheOrganizations([organization]); + } } // Checks whether organization exists. diff --git a/src/resolvers/Mutation/adminRemoveEvent.ts b/src/resolvers/Mutation/adminRemoveEvent.ts index a2b17f472f..81757b749e 100644 --- a/src/resolvers/Mutation/adminRemoveEvent.ts +++ b/src/resolvers/Mutation/adminRemoveEvent.ts @@ -1,17 +1,18 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { adminCheck } from "../../utilities"; -import { User, Organization, Event } from "../../models"; import { - USER_NOT_FOUND_ERROR, - ORGANIZATION_NOT_FOUND_ERROR, EVENT_NOT_FOUND_ERROR, + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; -import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, Event, Organization, User } from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; import { deleteEventFromCache } from "../../services/EventCache/deleteEventFromCache"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { adminCheck } from "../../utilities"; /** * This function enables an admin to remove a event * @param _parent - parent of current request @@ -65,8 +66,9 @@ export const adminRemoveEvent: MutationResolvers["adminRemoveEvent"] = async ( organization = await Organization.findOne({ _id: event.organization, }).lean(); - - await cacheOrganizations([organization!]); + if (organization !== null) { + await cacheOrganizations([organization]); + } } // Checks whether organization exists. @@ -90,6 +92,16 @@ export const adminRemoveEvent: MutationResolvers["adminRemoveEvent"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Checks whether currentUser is an admin of organization. await adminCheck(currentUser._id, organization); @@ -102,15 +114,23 @@ export const adminRemoveEvent: MutationResolvers["adminRemoveEvent"] = async ( { _id: currentUser._id, }, + { + $pull: { + registeredEvents: event._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: currentUser._id, + }, { $pull: { eventAdmin: event._id, createdEvents: event._id, - registeredEvents: event._id, }, } ); - // Deletes the event. await Event.deleteOne({ _id: event._id, diff --git a/src/resolvers/Mutation/adminRemoveGroup.ts b/src/resolvers/Mutation/adminRemoveGroup.ts index 0769b10f7f..3019fd6713 100644 --- a/src/resolvers/Mutation/adminRemoveGroup.ts +++ b/src/resolvers/Mutation/adminRemoveGroup.ts @@ -1,11 +1,12 @@ import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import { adminCheck } from "../../utilities"; -import { User, Organization, GroupChat } from "../../models"; +import { User, Organization, GroupChat, AppUserProfile } from "../../models"; import { USER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, CHAT_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, } from "../../constants"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; @@ -52,8 +53,9 @@ export const adminRemoveGroup: MutationResolvers["adminRemoveGroup"] = async ( organization = await Organization.findOne({ _id: groupChat.organization, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) { + await cacheOrganizations([organization]); + } } // Checks whether organization exists. @@ -77,6 +79,16 @@ export const adminRemoveGroup: MutationResolvers["adminRemoveGroup"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: context.userId, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Checks whether currentUser with _id === context.userId is an admin of organization. await adminCheck(context.userId, organization); diff --git a/src/resolvers/Mutation/assignUserTag.ts b/src/resolvers/Mutation/assignUserTag.ts index 1cd81c67c5..84736fc4e3 100644 --- a/src/resolvers/Mutation/assignUserTag.ts +++ b/src/resolvers/Mutation/assignUserTag.ts @@ -1,13 +1,35 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { User, OrganizationTagUser, TagUser } from "../../models"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, TAG_NOT_FOUND, - USER_DOES_NOT_BELONG_TO_TAGS_ORGANIZATION, USER_ALREADY_HAS_TAG, + USER_DOES_NOT_BELONG_TO_TAGS_ORGANIZATION, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { + AppUserProfile, + OrganizationTagUser, + TagUser, + User, +} from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; + +/** + * This function enables an admin to assign tag to user or not. + * @param _parent - parent of current request + * @param args - payload provided with the request + * @param context - context of entire application + * @remarks The following checks are done: + * 1. If the user exists. + * 2. If the user has appProfile. + * 3. If the tag object exists. + * 4. If the user is an admin for the organization. + * 5. If the user to be assigned the tag exists. + * 6. If the user to be assigned the tag belongs to the tag's organization. + * 7. If the user already has the tag. + * @returns User to which the tag is assigned. + */ export const assignUserTag: MutationResolvers["assignUserTag"] = async ( _parent, @@ -26,6 +48,17 @@ export const assignUserTag: MutationResolvers["assignUserTag"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Get the tag object const tag = await OrganizationTagUser.findOne({ @@ -41,15 +74,13 @@ export const assignUserTag: MutationResolvers["assignUserTag"] = async ( } // Boolean to determine whether user is an admin of organization of the tag. - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(tag.organizationId) - ); - // Checks whether currentUser can assign the tag or not. - if ( - !currentUserIsOrganizationAdmin && - !(currentUser.userType === "SUPERADMIN") - ) { + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (orgId) => + orgId && Types.ObjectId(orgId.toString()).equals(tag.organizationId) + ); + //check whether current user can assign tag to user or not + if (!(currentUserIsOrganizationAdmin || currentUserAppProfile.isSuperAdmin)) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, diff --git a/src/resolvers/Mutation/blockPluginCreationBySuperadmin.ts b/src/resolvers/Mutation/blockPluginCreationBySuperadmin.ts index b196752abb..f22f6fd51c 100644 --- a/src/resolvers/Mutation/blockPluginCreationBySuperadmin.ts +++ b/src/resolvers/Mutation/blockPluginCreationBySuperadmin.ts @@ -1,7 +1,10 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../constants"; import { errors, requestContext } from "../../libraries"; -import { User } from "../../models"; -import { USER_NOT_FOUND_ERROR } from "../../constants"; +import { AppUserProfile, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { superAdminCheck } from "../../utilities"; /** * This function enables an admin to create block plugin. @@ -10,6 +13,7 @@ import { superAdminCheck } from "../../utilities"; * @param context - context of entire application * @remarks The following checks are done: * 1. If the user exists + * 2.If the user has appUserProfile * 2. If the user is the SUPERADMIN of organization * @returns Deleted updated user */ @@ -27,17 +31,37 @@ export const blockPluginCreationBySuperadmin: MutationResolvers["blockPluginCrea USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); - // Checks whether currentUser is a SUPERADMIN. - superAdminCheck(currentUser); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + // Checks whether currentUser is a SUPERADMIN. + superAdminCheck(currentUserAppProfile); + const userAppProfile = await AppUserProfile.findOne({ + userId: args.userId, + }).lean(); + if (!userAppProfile) { + throw new errors.NotFoundError( + requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); + } /* - Sets pluginCreationAllowed field on document of user with _id === args.userId + Sets pluginCreationAllowed field on document of appUserProfile with _id === args.userId to !args.blockUser and returns the updated user. */ - return await User.findOneAndUpdate( + return await AppUserProfile.findOneAndUpdate( { - _id: args.userId, + _id: userAppProfile._id, }, { $set: { diff --git a/src/resolvers/Mutation/blockUser.ts b/src/resolvers/Mutation/blockUser.ts index 9c3321bf9f..dd16e6cda7 100644 --- a/src/resolvers/Mutation/blockUser.ts +++ b/src/resolvers/Mutation/blockUser.ts @@ -1,17 +1,17 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { adminCheck } from "../../utilities"; +import { Types } from "mongoose"; import { - USER_NOT_AUTHORIZED_ERROR, - ORGANIZATION_NOT_FOUND_ERROR, MEMBER_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, + ORGANIZATION_NOT_FOUND_ERROR, USER_BLOCKING_SELF, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; import { Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; -import { Types } from "mongoose"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { adminCheck } from "../../utilities"; /** * This function enables blocking a user. * @param _parent - parent of current request @@ -41,8 +41,9 @@ export const blockUser: MutationResolvers["blockUser"] = async ( organization = await Organization.findOne({ _id: args.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) { + await cacheOrganizations([organization]); + } } // Checks whether organization exists. diff --git a/src/resolvers/Mutation/cancelMembershipRequest.ts b/src/resolvers/Mutation/cancelMembershipRequest.ts index b48d4a4930..6f5b7d6a7c 100644 --- a/src/resolvers/Mutation/cancelMembershipRequest.ts +++ b/src/resolvers/Mutation/cancelMembershipRequest.ts @@ -1,14 +1,14 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, Organization, MembershipRequest } from "../../models"; -import { errors, requestContext } from "../../libraries"; import { + MEMBERSHIP_REQUEST_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, - MEMBERSHIP_REQUEST_NOT_FOUND_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import { errors, requestContext } from "../../libraries"; +import { MembershipRequest, Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to cancel membership request. * @param _parent - parent of current request @@ -48,8 +48,9 @@ export const cancelMembershipRequest: MutationResolvers["cancelMembershipRequest organization = await Organization.findOne({ _id: membershipRequest.organization, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) { + await cacheOrganizations([organization]); + } } // Checks whether organization exists. diff --git a/src/resolvers/Mutation/checkIn.ts b/src/resolvers/Mutation/checkIn.ts index 662d8cb704..e637ed5462 100644 --- a/src/resolvers/Mutation/checkIn.ts +++ b/src/resolvers/Mutation/checkIn.ts @@ -1,17 +1,23 @@ +import { Types } from "mongoose"; import { EVENT_NOT_FOUND_ERROR, + USER_ALREADY_CHECKED_IN, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, USER_NOT_REGISTERED_FOR_EVENT, - USER_ALREADY_CHECKED_IN, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import type { InterfaceEvent } from "../../models"; -import { User, Event, EventAttendee, CheckIn } from "../../models"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { + AppUserProfile, + CheckIn, + Event, + EventAttendee, + User, +} from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; -import { Types } from "mongoose"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const checkIn: MutationResolvers["checkIn"] = async ( _parent, @@ -29,7 +35,16 @@ export const checkIn: MutationResolvers["checkIn"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } - + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let currentEvent: InterfaceEvent | null; const eventFoundInCache = await findEventsInCache([args.data.eventId]); @@ -59,7 +74,7 @@ export const checkIn: MutationResolvers["checkIn"] = async ( admin === context.userID || Types.ObjectId(admin).equals(context.userId) ); - if (!isUserEventAdmin && currentUser.userType !== "SUPERADMIN") { + if (!isUserEventAdmin && currentUserAppProfile.isSuperAdmin === false) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, @@ -101,7 +116,7 @@ export const checkIn: MutationResolvers["checkIn"] = async ( } const checkIn = await CheckIn.create({ - eventAttendeeId: attendeeData!._id, + eventAttendeeId: attendeeData._id, allotedSeat: args.data.allotedSeat, allotedRoom: args.data.allotedRoom, }); diff --git a/src/resolvers/Mutation/createActionItem.ts b/src/resolvers/Mutation/createActionItem.ts index c2ec83ff93..75b5c88d35 100644 --- a/src/resolvers/Mutation/createActionItem.ts +++ b/src/resolvers/Mutation/createActionItem.ts @@ -1,17 +1,23 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import type { InterfaceActionItem, InterfaceEvent } from "../../models"; -import { User, Event, ActionItemCategory, ActionItem } from "../../models"; -import { errors, requestContext } from "../../libraries"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, - EVENT_NOT_FOUND_ERROR, ACTION_ITEM_CATEGORY_NOT_FOUND_ERROR, + EVENT_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, USER_NOT_MEMBER_FOR_ORGANIZATION, } from "../../constants"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceActionItem, InterfaceEvent } from "../../models"; +import { + ActionItem, + ActionItemCategory, + AppUserProfile, + Event, + User, +} from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; -import { Types } from "mongoose"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to create an action item. @@ -20,6 +26,7 @@ import { Types } from "mongoose"; * @param context - context of entire application * @remarks The following checks are done: * 1. If the user exists + * 2.If the user has appUserProfile * 3. If the asignee exists * 4. If the actionItemCategory exists * 5. If the asignee is a member of the organization @@ -46,6 +53,16 @@ export const createActionItem: MutationResolvers["createActionItem"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const assignee = await User.findOne({ _id: args.data.assigneeId, @@ -125,17 +142,20 @@ export const createActionItem: MutationResolvers["createActionItem"] = async ( } // Checks if the currUser is an admin of the organization - const currentUserIsOrgAdmin = currentUser.adminFor.some( - (ogranizationId) => - ogranizationId === actionItemCategory.organizationId || - Types.ObjectId(ogranizationId).equals(actionItemCategory.organizationId) + const currentUserIsOrgAdmin = currentUserAppProfile.adminFor.some( + (organizationId) => + (organizationId && + organizationId === actionItemCategory.organizationId) || + Types.ObjectId(organizationId?.toString()).equals( + actionItemCategory.organizationId + ) ); // Checks whether currentUser with _id === context.userId is authorized for the operation. if ( currentUserIsEventAdmin === false && currentUserIsOrgAdmin === false && - currentUser.userType !== "SUPERADMIN" + !currentUserAppProfile.isSuperAdmin ) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), diff --git a/src/resolvers/Mutation/createActionItemCategory.ts b/src/resolvers/Mutation/createActionItemCategory.ts index 2df722f91a..73f5e117ff 100644 --- a/src/resolvers/Mutation/createActionItemCategory.ts +++ b/src/resolvers/Mutation/createActionItemCategory.ts @@ -1,15 +1,15 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, ActionItemCategory, Organization } from "../../models"; -import { errors, requestContext } from "../../libraries"; import { - USER_NOT_FOUND_ERROR, - ORGANIZATION_NOT_FOUND_ERROR, ACTION_ITEM_CATEGORY_ALREADY_EXISTS, + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { ActionItemCategory, Organization, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { adminCheck } from "../../utilities"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import { adminCheck } from "../../utilities"; /** * This function enables to create an ActionItemCategory. @@ -51,8 +51,9 @@ export const createActionItemCategory: MutationResolvers["createActionItemCatego organization = await Organization.findOne({ _id: args.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) { + await cacheOrganizations([organization]); + } } // Checks whether the organization with _id === args.organizationId exists. diff --git a/src/resolvers/Mutation/createAdmin.ts b/src/resolvers/Mutation/createAdmin.ts index 8a558a76ff..4a3181c381 100644 --- a/src/resolvers/Mutation/createAdmin.ts +++ b/src/resolvers/Mutation/createAdmin.ts @@ -1,16 +1,17 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, Organization } from "../../models"; -import { errors, requestContext } from "../../libraries"; -import { superAdminCheck } from "../../utilities"; +import { Types } from "mongoose"; import { - ORGANIZATION_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, ORGANIZATION_MEMBER_NOT_FOUND_ERROR, + ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceOrganization } from "../../models"; +import { AppUserProfile, Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; -import { Types } from "mongoose"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { superAdminCheck } from "../../utilities"; /** * This function enables to create an admin for an organization. * @param _parent - parent of current request @@ -18,11 +19,12 @@ import { Types } from "mongoose"; * @param context - context of entire application * @remarks The following checks are done: * 1. If the organization exists - * 2. If the current user is the creator of the organization - * 3. If the user exists - * 4. If the user is a member of the organization - * 4. If the user is already an admin of the organization - * @returns Updated user + * 2. If the user has appUserProfile + * 3. If the current user is the creator of the organization + * 4. If the user exists + * 5. If the user is a member of the organization + * 6. If the user is already an admin of the organization + * @returns Updated appUserProfile */ export const createAdmin: MutationResolvers["createAdmin"] = async ( _parent, @@ -42,7 +44,7 @@ export const createAdmin: MutationResolvers["createAdmin"] = async ( _id: args.data.organizationId, }).lean(); - await cacheOrganizations([organization!]); + await cacheOrganizations([organization as InterfaceOrganization]); } // Checks whether organization exists. @@ -64,8 +66,30 @@ export const createAdmin: MutationResolvers["createAdmin"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } - superAdminCheck(currentUser); + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + + superAdminCheck(currentUserAppProfile); + const userAppProfile = await AppUserProfile.findOne({ + userId: args.data.userId, + }).lean(); + 1; + if (!userAppProfile) { + throw new errors.NotFoundError( + requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); + } const userExists = await User.exists({ _id: args.data.userId, }); @@ -125,12 +149,12 @@ export const createAdmin: MutationResolvers["createAdmin"] = async ( } /* - Adds organization._id to adminFor list on user's document with _id === args.data.userId - and returns the updated user. + Adds organization._id to adminFor list on appUserProfile's document with userId === args.data.userId + and returns the updated appUserProfile of the user. */ - return await User.findOneAndUpdate( + return await AppUserProfile.findOneAndUpdate( { - _id: args.data.userId, + _id: userAppProfile._id, }, { $push: { @@ -140,7 +164,5 @@ export const createAdmin: MutationResolvers["createAdmin"] = async ( { new: true, } - ) - .select(["-password"]) - .lean(); + ).lean(); }; diff --git a/src/resolvers/Mutation/createAdvertisement.ts b/src/resolvers/Mutation/createAdvertisement.ts index d3e4bb7c92..434b2bf26f 100644 --- a/src/resolvers/Mutation/createAdvertisement.ts +++ b/src/resolvers/Mutation/createAdvertisement.ts @@ -1,6 +1,6 @@ import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { Advertisement } from "../../models"; -// @ts-ignore + export const createAdvertisement: MutationResolvers["createAdvertisement"] = // eslint-disable-next-line @typescript-eslint/no-unused-vars async (_parent, args, _context) => { diff --git a/src/resolvers/Mutation/createDirectChat.ts b/src/resolvers/Mutation/createDirectChat.ts index 9c32226364..5d7d6daf7e 100644 --- a/src/resolvers/Mutation/createDirectChat.ts +++ b/src/resolvers/Mutation/createDirectChat.ts @@ -34,8 +34,7 @@ export const createDirectChat: MutationResolvers["createDirectChat"] = async ( organization = await Organization.findOne({ _id: args.data.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks whether organization with _id === args.data.organizationId exists. diff --git a/src/resolvers/Mutation/createEvent.ts b/src/resolvers/Mutation/createEvent.ts index d05e10a43e..3ec59b5d6e 100644 --- a/src/resolvers/Mutation/createEvent.ts +++ b/src/resolvers/Mutation/createEvent.ts @@ -1,20 +1,22 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import type { InterfaceEvent, InterfaceUser } from "../../models"; -import { User, Organization } from "../../models"; +import type mongoose from "mongoose"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, - ORGANIZATION_NOT_FOUND_ERROR, - ORGANIZATION_NOT_AUTHORIZED_ERROR, LENGTH_VALIDATION_ERROR, + ORGANIZATION_NOT_AUTHORIZED_ERROR, + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; -import { isValidString } from "../../libraries/validators/validateString"; +import { session } from "../../db"; +import { Once, Weekly } from "../../helpers/eventInstances"; +import { errors, requestContext } from "../../libraries"; import { compareDates } from "../../libraries/validators/compareDates"; +import { isValidString } from "../../libraries/validators/validateString"; +import type { InterfaceEvent, InterfaceUser } from "../../models"; +import { AppUserProfile, Organization, User } from "../../models"; import { EventAttendee } from "../../models/EventAttendee"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; -import type mongoose from "mongoose"; -import { session } from "../../db"; -import { Weekly, Once } from "../../helpers/eventInstances"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to create an event. @@ -23,8 +25,9 @@ import { Weekly, Once } from "../../helpers/eventInstances"; * @param context - context of entire application * @remarks The following checks are done: * 1. If the user exists - * 2. If the organization exists - * 3. If the user is a part of the organization. + * 2.If the user has appUserProfile + * 3. If the organization exists + * 4. If the user is a part of the organization. * @returns Created event */ export const createEvent: MutationResolvers["createEvent"] = async ( @@ -44,6 +47,17 @@ export const createEvent: MutationResolvers["createEvent"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const organization = await Organization.findOne({ _id: args.data?.organizationId, @@ -58,9 +72,10 @@ export const createEvent: MutationResolvers["createEvent"] = async ( ); } - const userCreatedOrganization = currentUser.createdOrganizations.some( - (createdOrganization) => createdOrganization.equals(organization._id) - ); + const userCreatedOrganization = + currentUserAppProfile.createdOrganizations.some((createdOrganization) => + Types.ObjectId(createdOrganization?.toString()).equals(organization._id) + ); const userJoinedOrganization = currentUser.joinedOrganizations.some( (joinedOrganization) => joinedOrganization.equals(organization._id) @@ -71,7 +86,7 @@ export const createEvent: MutationResolvers["createEvent"] = async ( !( userCreatedOrganization || userJoinedOrganization || - currentUser.userType == "SUPERADMIN" + currentUserAppProfile.isSuperAdmin ) ) { throw new errors.UnauthorizedError( @@ -204,16 +219,26 @@ async function associateEventWithUser( ], { session } ); - await User.updateOne( { _id: currentUser._id, }, + { + $push: { + registeredEvents: createdEvent._id, + }, + }, + { session } + ); + + await AppUserProfile.updateOne( + { + user: currentUser._id, + }, { $push: { eventAdmin: createdEvent._id, createdEvents: createdEvent._id, - registeredEvents: createdEvent._id, }, }, { session } diff --git a/src/resolvers/Mutation/createGroupChat.ts b/src/resolvers/Mutation/createGroupChat.ts index 53835f8d45..e7d5eeaa68 100644 --- a/src/resolvers/Mutation/createGroupChat.ts +++ b/src/resolvers/Mutation/createGroupChat.ts @@ -1,12 +1,12 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, GroupChat, Organization } from "../../models"; -import { errors, requestContext } from "../../libraries"; import { - USER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { GroupChat, Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to create a group chat. * @param _parent - parent of current request @@ -34,8 +34,7 @@ export const createGroupChat: MutationResolvers["createGroupChat"] = async ( organization = await Organization.findOne({ _id: args.data.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks whether organization with _id === args.data.organizationId exists. diff --git a/src/resolvers/Mutation/createMember.ts b/src/resolvers/Mutation/createMember.ts index d55623a899..a08d6b3170 100644 --- a/src/resolvers/Mutation/createMember.ts +++ b/src/resolvers/Mutation/createMember.ts @@ -1,14 +1,16 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { User, Organization } from "../../models"; -import { superAdminCheck } from "../../utilities"; import { + MEMBER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, - MEMBER_NOT_FOUND_ERROR, } from "../../constants"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceOrganization } from "../../models"; +import { AppUserProfile, Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { superAdminCheck } from "../../utilities"; /** * This function enables to add a member. * @param _parent - parent of current request @@ -18,7 +20,9 @@ import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrgani * 1. Checks whether current user making the request is an superAdmin * 2. If the organization exists * 3. Checks whether curent user exists. + * 4. Checks whether current user has appProfile. * 4. Checks whether user with _id === args.input.userId is already an member of organization.. + * * @returns Organization. */ export const createMember: MutationResolvers["createMember"] = async ( @@ -38,7 +42,17 @@ export const createMember: MutationResolvers["createMember"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } - superAdminCheck(currentUser); + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + superAdminCheck(currentUserAppProfile); // Checks if organization exists. let organization; @@ -54,7 +68,7 @@ export const createMember: MutationResolvers["createMember"] = async ( _id: args.input.organizationId, }).lean(); - await cacheOrganizations([organization!]); + await cacheOrganizations([organization as InterfaceOrganization]); } if (!organization) { @@ -125,5 +139,5 @@ export const createMember: MutationResolvers["createMember"] = async ( await cacheOrganizations([updatedOrganization]); } - return updatedOrganization!; + return updatedOrganization as InterfaceOrganization; }; diff --git a/src/resolvers/Mutation/createMessageChat.ts b/src/resolvers/Mutation/createMessageChat.ts index 5d828e9b4d..64332a6855 100644 --- a/src/resolvers/Mutation/createMessageChat.ts +++ b/src/resolvers/Mutation/createMessageChat.ts @@ -1,7 +1,10 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, MessageChat } from "../../models"; +import { + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../constants"; import { errors, requestContext } from "../../libraries"; -import { USER_NOT_FOUND_ERROR } from "../../constants"; +import { AppUserProfile, MessageChat, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to create a chat. * @param _parent - parent of current request @@ -10,6 +13,7 @@ import { USER_NOT_FOUND_ERROR } from "../../constants"; * @remarks The following checks are done: * 1. If the receiver user exists * 2. If the sender and receiver users have same language code. + * 3. If the sender and receiver users have appProfile. * @returns Created message chat. */ export const createMessageChat: MutationResolvers["createMessageChat"] = async ( @@ -20,6 +24,17 @@ export const createMessageChat: MutationResolvers["createMessageChat"] = async ( const currentUser = await User.findOne({ _id: context.userId, }).lean(); + if (!currentUser) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); + } + + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); const receiverUser = await User.findOne({ _id: args.data.receiver, @@ -33,10 +48,28 @@ export const createMessageChat: MutationResolvers["createMessageChat"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const receiverUserAppProfile = await AppUserProfile.findOne({ + userId: receiverUser._id, + }).lean(); + if (!receiverUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Boolean to identify whether both sender and receiver for messageChat have the same appLanguageCode. const isSenderReceiverLanguageSame = - receiverUser?.appLanguageCode === currentUser?.appLanguageCode; + receiverUserAppProfile?.appLanguageCode === + currentUserAppProfile?.appLanguageCode; // Creates new messageChat. const createdMessageChat = await MessageChat.create({ diff --git a/src/resolvers/Mutation/createOrganization.ts b/src/resolvers/Mutation/createOrganization.ts index 46ab307946..a71a93b181 100644 --- a/src/resolvers/Mutation/createOrganization.ts +++ b/src/resolvers/Mutation/createOrganization.ts @@ -1,15 +1,24 @@ import "dotenv/config"; +import { + LENGTH_VALIDATION_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { isValidString } from "../../libraries/validators/validateString"; +import { + ActionItemCategory, + AppUserProfile, + Organization, + User, +} from "../../models"; +import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import type { - MutationResolvers, Address, + MutationResolvers, } from "../../types/generatedGraphQLTypes"; -import { User, Organization, ActionItemCategory } from "../../models"; -import { errors, requestContext } from "../../libraries"; -import { LENGTH_VALIDATION_ERROR } from "../../constants"; import { superAdminCheck } from "../../utilities"; -import { isValidString } from "../../libraries/validators/validateString"; import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; -import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; /** * This function enables to create an organization. * @param _parent - parent of current request @@ -17,6 +26,7 @@ import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrgani * @param context - context of entire application * @remarks The following checks are done: * 1. If the user exists + * 2. If the user has appUserProfile * @returns Created organization */ export const createOrganization: MutationResolvers["createOrganization"] = @@ -25,9 +35,24 @@ export const createOrganization: MutationResolvers["createOrganization"] = _id: context.userId, }); - if (currentUser) { - superAdminCheck(currentUser); + if (!currentUser) { + throw new errors.NotFoundError( + requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + superAdminCheck(currentUserAppProfile); //Upload file let uploadImageFileName = null; @@ -98,6 +123,7 @@ export const createOrganization: MutationResolvers["createOrganization"] = Adds createdOrganization._id to joinedOrganizations, createdOrganizations and adminFor lists on currentUser's document with _id === context.userId */ + await User.updateOne( { _id: context.userId, @@ -105,6 +131,15 @@ export const createOrganization: MutationResolvers["createOrganization"] = { $push: { joinedOrganizations: createdOrganization._id, + }, + } + ); + await AppUserProfile.updateOne( + { + _id: currentUserAppProfile._id, + }, + { + $push: { createdOrganizations: createdOrganization._id, adminFor: createdOrganization._id, }, diff --git a/src/resolvers/Mutation/createPost.ts b/src/resolvers/Mutation/createPost.ts index 10f4b782ee..aaf884016c 100644 --- a/src/resolvers/Mutation/createPost.ts +++ b/src/resolvers/Mutation/createPost.ts @@ -1,20 +1,23 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, Post, Organization } from "../../models"; -import { errors, requestContext } from "../../libraries"; +import { Types } from "mongoose"; import { LENGTH_VALIDATION_ERROR, ORGANIZATION_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_TO_PIN, - POST_NEEDS_TO_BE_PINNED, PLEASE_PROVIDE_TITLE, + POST_NEEDS_TO_BE_PINNED, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_AUTHORIZED_TO_PIN, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, logger, requestContext } from "../../libraries"; import { isValidString } from "../../libraries/validators/validateString"; -import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; -import { uploadEncodedVideo } from "../../utilities/encodedVideoStorage/uploadEncodedVideo"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { InterfaceOrganization } from "../../models"; +import { AppUserProfile, Organization, Post, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; import { cachePosts } from "../../services/PostCache/cachePosts"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; +import { uploadEncodedVideo } from "../../utilities/encodedVideoStorage/uploadEncodedVideo"; /** * This function enables to create a post. * @param _parent - parent of current request @@ -23,6 +26,7 @@ import { cachePosts } from "../../services/PostCache/cachePosts"; * @remarks The following checks are done: * 1. If the user exists * 2. If the organization exists + * 3. If the user has appUserProfile * @returns Created Post */ export const createPost: MutationResolvers["createPost"] = async ( @@ -37,6 +41,7 @@ export const createPost: MutationResolvers["createPost"] = async ( // Checks whether currentUser exists. if (!currentUser) { + logger.info("here"); throw new errors.NotFoundError( requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), USER_NOT_FOUND_ERROR.CODE, @@ -44,6 +49,17 @@ export const createPost: MutationResolvers["createPost"] = async ( ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + let organization; const organizationFoundInCache = await findOrganizationsInCache([ @@ -57,7 +73,7 @@ export const createPost: MutationResolvers["createPost"] = async ( _id: args.data.organizationId, }).lean(); - await cacheOrganizations([organization!]); + await cacheOrganizations([organization as InterfaceOrganization]); } // Checks whether organization with _id == args.data.organizationId exists. @@ -120,20 +136,21 @@ export const createPost: MutationResolvers["createPost"] = async ( if (args.data.pinned) { // Check if the user has privileges to pin the post - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organizationId) => organizationId.equals(args.data.organizationId) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (organizationId) => + Types.ObjectId(organizationId?.toString()).equals( + args.data.organizationId + ) ); - if (currentUser?.userType) { - if ( - !(currentUser?.userType === "SUPERADMIN") && - !currentUserIsOrganizationAdmin - ) { - throw new errors.UnauthorizedError( - requestContext.translate(USER_NOT_AUTHORIZED_TO_PIN.MESSAGE), - USER_NOT_AUTHORIZED_TO_PIN.CODE, - USER_NOT_AUTHORIZED_TO_PIN.PARAM - ); - } + + if ( + !(currentUserAppProfile.isSuperAdmin || currentUserIsOrganizationAdmin) + ) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_TO_PIN.MESSAGE), + USER_NOT_AUTHORIZED_TO_PIN.CODE, + USER_NOT_AUTHORIZED_TO_PIN.PARAM + ); } } @@ -165,7 +182,7 @@ export const createPost: MutationResolvers["createPost"] = async ( } ); - await cacheOrganizations([updatedOrganizaiton!]); + await cacheOrganizations([updatedOrganizaiton as InterfaceOrganization]); } // Returns createdPost. diff --git a/src/resolvers/Mutation/createSampleOrganization.ts b/src/resolvers/Mutation/createSampleOrganization.ts index 86fd355f16..695b6ecfd8 100644 --- a/src/resolvers/Mutation/createSampleOrganization.ts +++ b/src/resolvers/Mutation/createSampleOrganization.ts @@ -1,12 +1,11 @@ import { USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, - SAMPLE_ORGANIZATION_ALREADY_EXISTS, } from "../../constants"; import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, User } from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { createSampleOrganization as createSampleOrgUtil } from "../../utilities/createSampleOrganizationUtil"; -import { SampleData, User } from "../../models"; /** * Generates sample data for testing or development purposes. @@ -26,12 +25,10 @@ export const createSampleOrganization: MutationResolvers["createSampleOrganizati ); } - if ( - !( - currentUser.userType === "SUPERADMIN" || - currentUser.userType === "ADMIN" - ) - ) { + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, @@ -39,15 +36,11 @@ export const createSampleOrganization: MutationResolvers["createSampleOrganizati ); } - const existingOrganization = await SampleData.findOne({ - collectionName: "Organization", - }); - - if (existingOrganization) { + if (!currentUserAppProfile.isSuperAdmin) { throw new errors.UnauthorizedError( - requestContext.translate(SAMPLE_ORGANIZATION_ALREADY_EXISTS.MESSAGE), - SAMPLE_ORGANIZATION_ALREADY_EXISTS.CODE, - SAMPLE_ORGANIZATION_ALREADY_EXISTS.PARAM + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM ); } diff --git a/src/resolvers/Mutation/createUserFamily.ts b/src/resolvers/Mutation/createUserFamily.ts index 9a7f911625..96dd44d47a 100644 --- a/src/resolvers/Mutation/createUserFamily.ts +++ b/src/resolvers/Mutation/createUserFamily.ts @@ -1,13 +1,16 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; -import { errors, requestContext } from "../../libraries"; import { LENGTH_VALIDATION_ERROR, USER_FAMILY_MIN_MEMBERS_ERROR_CODE, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../constants"; + +import { errors, requestContext } from "../../libraries"; import { isValidString } from "../../libraries/validators/validateString"; +import { AppUserProfile, User } from "../../models"; import { UserFamily } from "../../models/userFamily"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; + import { superAdminCheck } from "../../utilities"; /** * This Function enables to create a user Family @@ -38,8 +41,18 @@ export const createUserFamily: MutationResolvers["createUserFamily"] = async ( ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthenticatedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Check whether the user is super admin. - superAdminCheck(currentUser); + superAdminCheck(currentUserAppProfile); let validationResultName = { isLessThanMaxLength: false, diff --git a/src/resolvers/Mutation/createUserTag.ts b/src/resolvers/Mutation/createUserTag.ts index 6df7a899df..08629212eb 100644 --- a/src/resolvers/Mutation/createUserTag.ts +++ b/src/resolvers/Mutation/createUserTag.ts @@ -1,14 +1,21 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, OrganizationTagUser, Organization } from "../../models"; -import { errors, requestContext } from "../../libraries"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_TO_CREATE_TAG, INCORRECT_TAG_INPUT, ORGANIZATION_NOT_FOUND_ERROR, - TAG_NOT_FOUND, TAG_ALREADY_EXISTS, + TAG_NOT_FOUND, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_AUTHORIZED_TO_CREATE_TAG, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { + AppUserProfile, + Organization, + OrganizationTagUser, + User, +} from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const createUserTag: MutationResolvers["createUserTag"] = async ( _parent, @@ -28,6 +35,18 @@ export const createUserTag: MutationResolvers["createUserTag"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + //check whether current User has app profile or not + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Checks if the provided organization exists const organizationExists = await Organization.exists({ @@ -43,14 +62,14 @@ export const createUserTag: MutationResolvers["createUserTag"] = async ( } // Check if the user has privileges to create the tag - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organizationId) => organizationId.equals(args.input.organizationId) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (organizationId) => + Types.ObjectId(organizationId?.toString()).equals( + args.input.organizationId + ) ); - if ( - !((currentUser?.userType ?? "") === "SUPERADMIN") && - !currentUserIsOrganizationAdmin - ) { + if (!currentUserAppProfile.isSuperAdmin && !currentUserIsOrganizationAdmin) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_TO_CREATE_TAG.MESSAGE), USER_NOT_AUTHORIZED_TO_CREATE_TAG.CODE, diff --git a/src/resolvers/Mutation/index.ts b/src/resolvers/Mutation/index.ts index 2d3332bcf8..a58302664e 100644 --- a/src/resolvers/Mutation/index.ts +++ b/src/resolvers/Mutation/index.ts @@ -91,7 +91,7 @@ import { updatePost } from "./updatePost"; import { updateUserProfile } from "./updateUserProfile"; import { updateUserPassword } from "./updateUserPassword"; import { updateUserTag } from "./updateUserTag"; -import { updateUserType } from "./updateUserType"; + import { deleteAdvertisementById } from "./deleteAdvertisementById"; import { updateAdvertisement } from "./updateAdvertisement"; @@ -187,7 +187,7 @@ export const Mutation: MutationResolvers = { updatePluginStatus, updateUserProfile, updateUserPassword, - updateUserType, + updateUserTag, updatePost, updateAdvertisement, diff --git a/src/resolvers/Mutation/joinPublicOrganization.ts b/src/resolvers/Mutation/joinPublicOrganization.ts index 2f24dd36e9..e5b126908d 100644 --- a/src/resolvers/Mutation/joinPublicOrganization.ts +++ b/src/resolvers/Mutation/joinPublicOrganization.ts @@ -36,8 +36,7 @@ export const joinPublicOrganization: MutationResolvers["joinPublicOrganization"] organization = await Organization.findOne({ _id: args.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks whether organization exists. diff --git a/src/resolvers/Mutation/leaveOrganization.ts b/src/resolvers/Mutation/leaveOrganization.ts index d8b72ba98e..a2c83a9e12 100644 --- a/src/resolvers/Mutation/leaveOrganization.ts +++ b/src/resolvers/Mutation/leaveOrganization.ts @@ -1,14 +1,17 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User, Organization } from "../../models"; -import { errors, requestContext } from "../../libraries"; +import type { UpdateQuery } from "mongoose"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, MEMBER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceOrganization } from "../../models"; +import { AppUserProfile, Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; -import { Types } from "mongoose"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to leave an organization. * @param _parent - parent of current request @@ -38,8 +41,7 @@ export const leaveOrganization: MutationResolvers["leaveOrganization"] = async ( organization = await Organization.findOne({ _id: args.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks whether organization exists. @@ -63,6 +65,16 @@ export const leaveOrganization: MutationResolvers["leaveOrganization"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const currentUserIsOrganizationMember = organization.members.some((member) => Types.ObjectId(member).equals(currentUser?._id) @@ -76,18 +88,41 @@ export const leaveOrganization: MutationResolvers["leaveOrganization"] = async ( MEMBER_NOT_FOUND_ERROR.PARAM ); } + const currentUserIsOrgAdmin = organization.admins.some((admin) => + Types.ObjectId(admin).equals(currentUser._id) + ); // Removes currentUser._id from admins and members lists of organzation's document. - const updatedOrganization = await Organization.findOneAndUpdate( - { - _id: organization._id, + + let updateQuery: UpdateQuery = { + $pull: { + members: currentUser._id, }, - { + }; + if (currentUserIsOrgAdmin) { + await AppUserProfile.updateOne( + { + userId: currentUser._id, + }, + { + $pull: { + organizations: organization._id, + }, + } + ); + updateQuery = { $pull: { - admins: currentUser._id, members: currentUser._id, + admins: currentUser._id, }, + }; + } + + const updatedOrganization = await Organization.findOneAndUpdate( + { + _id: organization._id, }, + updateQuery, { new: true, } @@ -100,6 +135,7 @@ export const leaveOrganization: MutationResolvers["leaveOrganization"] = async ( Removes organization._id from joinedOrganizations list of currentUser's document and returns the updated currentUser. */ + return await User.findOneAndUpdate( { _id: currentUser._id, @@ -107,7 +143,6 @@ export const leaveOrganization: MutationResolvers["leaveOrganization"] = async ( { $pull: { joinedOrganizations: organization._id, - adminFor: organization._id, }, }, { diff --git a/src/resolvers/Mutation/login.ts b/src/resolvers/Mutation/login.ts index 94ae28496e..42612709cd 100644 --- a/src/resolvers/Mutation/login.ts +++ b/src/resolvers/Mutation/login.ts @@ -1,18 +1,22 @@ import bcrypt from "bcryptjs"; +import { + INVALID_CREDENTIALS_ERROR, + LAST_RESORT_SUPERADMIN_EMAIL, + USER_NOT_FOUND_ERROR, +} from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import type { + InterfaceEvent, + InterfaceOrganization, + InterfaceUser, +} from "../../models"; +import { AppUserProfile, User } from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import type { InterfaceUser } from "../../models"; -import { User } from "../../models"; import { + copyToClipboard, createAccessToken, createRefreshToken, - copyToClipboard, } from "../../utilities"; -import { errors, requestContext } from "../../libraries"; -import { - INVALID_CREDENTIALS_ERROR, - USER_NOT_FOUND_ERROR, - LAST_RESORT_SUPERADMIN_EMAIL, -} from "../../constants"; /** * This function enables login. * @param _parent - parent of current request @@ -35,9 +39,10 @@ export const login: MutationResolvers["login"] = async (_parent, args) => { USER_NOT_FOUND_ERROR.PARAM ); } + // console.log(user); const isPasswordValid = await bcrypt.compare( args.data.password, - user.password + user.password as string ); // Checks whether password is invalid. if (isPasswordValid === false) { @@ -52,8 +57,28 @@ export const login: MutationResolvers["login"] = async (_parent, args) => { requestContext.translate(INVALID_CREDENTIALS_ERROR.MESSAGE) ); } - const accessToken = await createAccessToken(user); - const refreshToken = await createRefreshToken(user); + let appUserProfile = await AppUserProfile.findOne({ + userId: user._id.toString(), + }).lean(); + if (!appUserProfile) { + appUserProfile = await AppUserProfile.create({ + userId: user._id.toString(), + appLanguageCode: "en", + tokenVersion: 0, + isSuperAdmin: false, + }); + await User.updateOne( + { + _id: user._id.toString(), + }, + { + appUserProfileId: appUserProfile._id.toString(), + } + ); + } + + const accessToken = createAccessToken(user, appUserProfile); + const refreshToken = createRefreshToken(user, appUserProfile); copyToClipboard(`{ "Authorization": "Bearer ${accessToken}" }`); @@ -61,40 +86,71 @@ export const login: MutationResolvers["login"] = async (_parent, args) => { // Updates the user to SUPERADMIN if the email of the user matches the LAST_RESORT_SUPERADMIN_EMAIL if ( user?.email.toLowerCase() === LAST_RESORT_SUPERADMIN_EMAIL?.toLowerCase() && - user?.userType !== "SUPERADMIN" + !appUserProfile.isSuperAdmin ) { - await User.updateOne( + // await User.updateOne( + // { + // _id: user?._id, + // }, + // { + // userType: "SUPERADMIN", + // } + // ); + await AppUserProfile.findOneAndUpdate( { - _id: user?._id, + user: user._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, } ); } - await User.findOneAndUpdate( - { _id: user._id }, - { token: refreshToken, $inc: { tokenVersion: 1 } } + // await User.findOneAndUpdate( + // { _id: user._id }, + // { token: refreshToken, $inc: { tokenVersion: 1 } } + // ); + await AppUserProfile.findOneAndUpdate( + { + user: user._id, + }, + { + token: refreshToken, + $inc: { + tokenVersion: 1, + }, + } ); - // Assigns new value with populated fields to user object. user = await User.findOne({ - _id: user._id, + _id: user._id.toString(), }) .select(["-password"]) .populate("joinedOrganizations") - .populate("createdOrganizations") - .populate("createdEvents") + // .populate("createdOrganizations") + // .populate("createdEvents") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + // .populate("eventAdmin") + // .populate("adminFor") .populate("membershipRequests") .populate("organizationsBlockedBy") .lean(); return { - user: user ?? ({} as InterfaceUser), + user: user as InterfaceUser, + appUserProfile: { + _id: appUserProfile._id.toString(), + userId: appUserProfile.userId as InterfaceUser, + adminFor: appUserProfile.adminFor as InterfaceOrganization[], + appLanguageCode: appUserProfile.appLanguageCode, + isSuperAdmin: appUserProfile.isSuperAdmin, + pluginCreationAllowed: appUserProfile.pluginCreationAllowed, + tokenVersion: appUserProfile.tokenVersion, + eventAdmin: appUserProfile.eventAdmin as InterfaceEvent[], + createdEvents: appUserProfile.createdEvents as InterfaceEvent[], + createdOrganizations: + appUserProfile.createdOrganizations as InterfaceOrganization[], + }, accessToken, refreshToken, }; diff --git a/src/resolvers/Mutation/logout.ts b/src/resolvers/Mutation/logout.ts index e4a3710e19..c84a9f0921 100644 --- a/src/resolvers/Mutation/logout.ts +++ b/src/resolvers/Mutation/logout.ts @@ -1,5 +1,5 @@ +import { AppUserProfile } from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; /** * This function enables logout. @@ -16,9 +16,9 @@ export const logout: MutationResolvers["logout"] = async ( context ) => { // Sets token field of currentUser with _id === context.userId to null. - await User.updateOne( + await AppUserProfile.updateOne( { - _id: context.userId, + userId: context.userId, }, { $set: { diff --git a/src/resolvers/Mutation/refreshToken.ts b/src/resolvers/Mutation/refreshToken.ts index ad98d6ccd5..6333dd55ef 100644 --- a/src/resolvers/Mutation/refreshToken.ts +++ b/src/resolvers/Mutation/refreshToken.ts @@ -1,18 +1,19 @@ import jwt from "jsonwebtoken"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { + INVALID_REFRESH_TOKEN_ERROR, + REFRESH_TOKEN_SECRET, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../constants"; import { errors, requestContext } from "../../libraries"; -import { User } from "../../models"; +import { AppUserProfile, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import type { InterfaceJwtTokenPayload } from "../../utilities"; import { createAccessToken, createRefreshToken, revokeRefreshToken, } from "../../utilities"; -import { - INVALID_REFRESH_TOKEN_ERROR, - REFRESH_TOKEN_SECRET, - USER_NOT_FOUND_ERROR, -} from "../../constants"; /** * This function creates a new access and refresh token. @@ -58,10 +59,20 @@ export const refreshToken: MutationResolvers["refreshToken"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const appUserProfile = await AppUserProfile.findOne({ + userId: user._id, + }).lean(); + if (!appUserProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } if ( - user.tokenVersion !== jwtPayload.tokenVersion && - user.token !== args.refreshToken + appUserProfile.tokenVersion !== jwtPayload.tokenVersion && + appUserProfile.token !== args.refreshToken ) { await revokeRefreshToken(jwtPayload.userId); throw new errors.ValidationError( @@ -79,11 +90,11 @@ export const refreshToken: MutationResolvers["refreshToken"] = async ( } // send new access and refresh token to user - const newAccessToken = await createAccessToken(user); - const newRefreshToken = await createRefreshToken(user); + const newAccessToken = await createAccessToken(user, appUserProfile); + const newRefreshToken = await createRefreshToken(user, appUserProfile); //update the token version for the user - const filter = { _id: jwtPayload.userId }; + const filter = { userId: jwtPayload.userId }; const update = { $set: { token: newRefreshToken, @@ -91,7 +102,7 @@ export const refreshToken: MutationResolvers["refreshToken"] = async ( $inc: { tokenVersion: 1 }, }; - await User.findOneAndUpdate(filter, update, { + await AppUserProfile.findOneAndUpdate(filter, update, { new: true, }); diff --git a/src/resolvers/Mutation/rejectAdmin.ts b/src/resolvers/Mutation/rejectAdmin.ts index cbf4073b4e..63c789b9fb 100644 --- a/src/resolvers/Mutation/rejectAdmin.ts +++ b/src/resolvers/Mutation/rejectAdmin.ts @@ -1,7 +1,10 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; -import { USER_NOT_FOUND_ERROR } from "../../constants"; +import { + USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, +} from "../../constants"; import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { superAdminCheck } from "../../utilities"; /** * This function enables to reject an admin. @@ -10,8 +13,9 @@ import { superAdminCheck } from "../../utilities"; * @param context - context of entire application * @remarks The following checks are done: * 1. If the user exists - * 2. If the user is the SUPERADMIN of the organization. - * 3. If the user to be removed exists. + * 2.If the user has appProfile or not (if not, then the user is not a superadmin). + * 3. If the user is the SUPERADMIN of the organization. + * 4. If the user to be removed exists. * @returns True if the operation is successful. */ export const rejectAdmin: MutationResolvers["rejectAdmin"] = async ( @@ -31,9 +35,19 @@ export const rejectAdmin: MutationResolvers["rejectAdmin"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); - // Checks whether currentUser is not a SUPERADMIN. - superAdminCheck(currentUser); + // if user does not have appProfile then he is NON_USER + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE), + USER_NOT_AUTHORIZED_SUPERADMIN.CODE, + USER_NOT_AUTHORIZED_SUPERADMIN.PARAM + ); + } + superAdminCheck(currentUserAppProfile); const userExists = await User.exists({ _id: args.id, diff --git a/src/resolvers/Mutation/removeActionItem.ts b/src/resolvers/Mutation/removeActionItem.ts index ae1c023213..9372f16152 100644 --- a/src/resolvers/Mutation/removeActionItem.ts +++ b/src/resolvers/Mutation/removeActionItem.ts @@ -1,16 +1,16 @@ +import { Types } from "mongoose"; import { ACTION_ITEM_NOT_FOUND_ERROR, EVENT_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import type { InterfaceEvent } from "../../models"; -import { User, ActionItem, Event } from "../../models"; -import { Types } from "mongoose"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { ActionItem, AppUserProfile, Event, User } from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to remove an action item. * @param _parent - parent of current request @@ -20,6 +20,7 @@ import { cacheEvents } from "../../services/EventCache/cacheEvents"; * 1. If the user exists. * 2. If the action item exists. * 3. If the user is authorized. + * 4. If the user has appUserProfile. * @returns deleted action item. */ @@ -40,6 +41,17 @@ export const removeActionItem: MutationResolvers["removeActionItem"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const actionItem = await ActionItem.findOne({ _id: args.id, @@ -56,10 +68,10 @@ export const removeActionItem: MutationResolvers["removeActionItem"] = async ( ); } - const currentUserIsOrgAdmin = currentUser.adminFor.some( + const currentUserIsOrgAdmin = currentUserAppProfile.adminFor.some( (ogranizationId) => ogranizationId === actionItem.actionItemCategoryId.organizationId || - Types.ObjectId(ogranizationId).equals( + Types.ObjectId(ogranizationId?.toString()).equals( actionItem.actionItemCategoryId.organizationId ) ); @@ -103,7 +115,7 @@ export const removeActionItem: MutationResolvers["removeActionItem"] = async ( if ( currentUserIsEventAdmin === false && currentUserIsOrgAdmin === false && - currentUser.userType !== "SUPERADMIN" + currentUserAppProfile.isSuperAdmin === false ) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), diff --git a/src/resolvers/Mutation/removeAdmin.ts b/src/resolvers/Mutation/removeAdmin.ts index 68f2920920..6bfc037e73 100644 --- a/src/resolvers/Mutation/removeAdmin.ts +++ b/src/resolvers/Mutation/removeAdmin.ts @@ -1,16 +1,17 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { superAdminCheck } from "../../utilities"; -import type { InterfaceOrganization } from "../../models"; -import { User, Organization } from "../../models"; -import { errors, requestContext } from "../../libraries"; +import { Types } from "mongoose"; import { ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, USER_NOT_ORGANIZATION_ADMIN, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceOrganization } from "../../models"; +import { AppUserProfile, Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; -import { Types } from "mongoose"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { superAdminCheck } from "../../utilities"; /** * This function enables to remove an admin. * @param _parent - parent of current request @@ -21,7 +22,8 @@ import { Types } from "mongoose"; * 2. If the organization exists. * 3. If the user to be removed is an admin. * 4. If the user removing the admin is the creator of the organization - * @returns Updated user. + * 5 .If the current user and user has appUserProfile or not + * @returns Updated appUserProfile. */ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( _parent, @@ -38,8 +40,9 @@ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( organization = await Organization.findOne({ _id: args.data.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization != null) { + await cacheOrganizations([organization]); + } } else { organization = organizationFoundInCache[0]; } @@ -60,6 +63,13 @@ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( const currentUser = await User.findOne({ _id: context.userId, }); + if (!currentUser) { + throw new errors.NotFoundError( + USER_NOT_FOUND_ERROR.MESSAGE, + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); + } // Checks whether user exists. if (!user) { @@ -69,7 +79,27 @@ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const userAppProfile = await AppUserProfile.findOne({ + userId: user._id, + }).lean(); + if (!userAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Checks whether user is an admin of the organization. const userIsOrganizationAdmin = organization.admins.some((admin) => Types.ObjectId(admin).equals(user._id) @@ -84,7 +114,7 @@ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( } // Checks whether the current user is a superadmin. - superAdminCheck(currentUser!); + superAdminCheck(currentUserAppProfile); // Removes user._id from admins list of the organization. const updatedOrganization = await Organization.findOneAndUpdate( @@ -94,7 +124,7 @@ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( { $set: { admins: organization.admins.filter( - (admin) => admin.toString() !== user!._id.toString() + (admin) => admin.toString() !== user._id.toString() ), }, }, @@ -107,15 +137,16 @@ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( await cacheOrganizations([updatedOrganization]); } - // Removes organization._id from adminFor list of the user and returns the updated user. - return await User.findOneAndUpdate( + // Removes organization._id from adminFor list of the appUserProfile and returns the updated userProfile. + return await AppUserProfile.findOneAndUpdate( { - _id: user._id, + _id: userAppProfile._id, }, { $set: { - adminFor: user.adminFor.filter( + adminFor: userAppProfile.adminFor.filter( (adminForOrganization) => + adminForOrganization && adminForOrganization.toString() !== organization._id.toString() ), }, @@ -123,7 +154,5 @@ export const removeAdmin: MutationResolvers["removeAdmin"] = async ( { new: true, } - ) - .select(["-password"]) - .lean(); + ).lean(); }; diff --git a/src/resolvers/Mutation/removeAdvertisement.ts b/src/resolvers/Mutation/removeAdvertisement.ts index 9c9fa0059c..ff5ec2764e 100644 --- a/src/resolvers/Mutation/removeAdvertisement.ts +++ b/src/resolvers/Mutation/removeAdvertisement.ts @@ -2,27 +2,31 @@ import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import { Advertisement } from "../../models"; import { ADVERTISEMENT_NOT_FOUND_ERROR } from "../../constants"; +import mongoose from "mongoose"; -// @ts-ignore +// @ts-expect-error : This block intentionally ignores TypeScript checking for incomplete code. export const removeAdvertisement: MutationResolvers["removeAdvertisement"] = // eslint-disable-next-line @typescript-eslint/no-unused-vars async (_parent, args, _context) => { - const currentAd = await Advertisement.findOne({ - _id: args.id ? args.id : "", - }).lean(); + const myId = args.id ? args.id : ""; - if (!currentAd) { - throw new errors.NotFoundError( - requestContext.translate(ADVERTISEMENT_NOT_FOUND_ERROR.MESSAGE), - ADVERTISEMENT_NOT_FOUND_ERROR.CODE, - ADVERTISEMENT_NOT_FOUND_ERROR.PARAM - ); - } + if (mongoose.Types.ObjectId.isValid(myId)) { + const currentAd = await Advertisement.findOne({ + _id: myId, + }).lean(); + if (!currentAd) { + throw new errors.NotFoundError( + requestContext.translate(ADVERTISEMENT_NOT_FOUND_ERROR.MESSAGE), + ADVERTISEMENT_NOT_FOUND_ERROR.CODE, + ADVERTISEMENT_NOT_FOUND_ERROR.PARAM + ); + } - // Deletes the ad. - await Advertisement.deleteOne({ - _id: args.id ? args.id : "", - }); - // Returns deleted ad. - return currentAd; + // Deletes the ad. + await Advertisement.deleteOne({ + _id: myId, + }); + // Returns deleted ad. + return currentAd; + } }; diff --git a/src/resolvers/Mutation/removeComment.ts b/src/resolvers/Mutation/removeComment.ts index 48dd672ca3..cd88fe27b0 100644 --- a/src/resolvers/Mutation/removeComment.ts +++ b/src/resolvers/Mutation/removeComment.ts @@ -1,15 +1,16 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import type { InterfaceComment } from "../../models"; -import { User, Post, Comment } from "../../models"; -import { errors, requestContext } from "../../libraries"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, COMMENT_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; -import { findCommentsInCache } from "../../services/CommentCache/findCommentsInCache"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceComment } from "../../models"; +import { AppUserProfile, Comment, Post, User } from "../../models"; import { deleteCommentFromCache } from "../../services/CommentCache/deleteCommentFromCache"; +import { findCommentsInCache } from "../../services/CommentCache/findCommentsInCache"; import { cachePosts } from "../../services/PostCache/cachePosts"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to remove a comment. @@ -20,6 +21,7 @@ import { cachePosts } from "../../services/PostCache/cachePosts"; * 1. If the user exists * 2. If the comment exists. * 3. If the user is the creator of the organization. + * 4. If the user has appUserProfile * @returns Deleted comment. */ @@ -40,7 +42,16 @@ export const removeComment: MutationResolvers["removeComment"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } - + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let comment: InterfaceComment; const commentsFoundInCache = await findCommentsInCache([args.id]); @@ -64,13 +75,17 @@ export const removeComment: MutationResolvers["removeComment"] = async ( ); } - const isCurrentUserAdminOfOrganization = currentUser.adminFor.some( - (organization) => organization.equals(comment.postId.organization) + const isCurrentUserAdminOfOrganization = currentUserAppProfile.adminFor.some( + (organization) => + organization && + Types.ObjectId(organization.toString()).equals( + comment.postId.organization + ) ); // Checks whether currentUser with _id === context.userId has the authorization to delete the comment if ( - currentUser.userType !== "SUPERADMIN" && + !currentUserAppProfile.isSuperAdmin && !isCurrentUserAdminOfOrganization && comment.creatorId.toString() !== context.userId ) { @@ -85,7 +100,7 @@ export const removeComment: MutationResolvers["removeComment"] = async ( const updatedPost = await Post.findOneAndUpdate( { - _id: comment!.postId._id, + _id: comment.postId._id, }, { $inc: { diff --git a/src/resolvers/Mutation/removeDirectChat.ts b/src/resolvers/Mutation/removeDirectChat.ts index 9d38689027..39b33e3147 100644 --- a/src/resolvers/Mutation/removeDirectChat.ts +++ b/src/resolvers/Mutation/removeDirectChat.ts @@ -36,8 +36,7 @@ export const removeDirectChat: MutationResolvers["removeDirectChat"] = async ( organization = await Organization.findOne({ _id: args.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks whether organization exists. diff --git a/src/resolvers/Mutation/removeEvent.ts b/src/resolvers/Mutation/removeEvent.ts index 63322f847c..1f579ae9b3 100644 --- a/src/resolvers/Mutation/removeEvent.ts +++ b/src/resolvers/Mutation/removeEvent.ts @@ -1,14 +1,15 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import type { InterfaceEvent } from "../../models"; -import { User, Event, ActionItem } from "../../models"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, EVENT_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceEvent } from "../../models"; +import { ActionItem, AppUserProfile, Event, User } from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to remove an event. * @param _parent - parent of current request @@ -19,6 +20,7 @@ import { cacheEvents } from "../../services/EventCache/cacheEvents"; * 2. If the event exists * 3. If the user is an admin of the organization. * 4. If the user is an admin of the event. + * 5. If the user has appUserProfile * @returns Deleted event. */ export const removeEvent: MutationResolvers["removeEvent"] = async ( @@ -38,6 +40,16 @@ export const removeEvent: MutationResolvers["removeEvent"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let event: InterfaceEvent | null; @@ -65,8 +77,10 @@ export const removeEvent: MutationResolvers["removeEvent"] = async ( } // Boolean to determine whether user is an admin of organization. - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(event?.organization) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (organization) => + organization && + Types.ObjectId(organization.toString()).equals(event?.organization) ); // Boolean to determine whether user is an admin of event. @@ -79,7 +93,7 @@ export const removeEvent: MutationResolvers["removeEvent"] = async ( !( currentUserIsOrganizationAdmin || currentUserIsEventAdmin || - currentUser.userType === "SUPERADMIN" + currentUserAppProfile.isSuperAdmin ) ) { throw new errors.UnauthorizedError( @@ -89,7 +103,7 @@ export const removeEvent: MutationResolvers["removeEvent"] = async ( ); } - await User.updateMany( + await AppUserProfile.updateMany( { createdEvents: event._id, }, @@ -100,7 +114,7 @@ export const removeEvent: MutationResolvers["removeEvent"] = async ( } ); - await User.updateMany( + await AppUserProfile.updateMany( { eventAdmin: event._id, }, diff --git a/src/resolvers/Mutation/removeEventAttendee.ts b/src/resolvers/Mutation/removeEventAttendee.ts index 05e67ad2ab..fe62f7a06c 100644 --- a/src/resolvers/Mutation/removeEventAttendee.ts +++ b/src/resolvers/Mutation/removeEventAttendee.ts @@ -7,7 +7,7 @@ import { import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import type { InterfaceEvent } from "../../models"; -import { User, Event, EventAttendee } from "../../models"; +import { User, Event, EventAttendee, AppUserProfile } from "../../models"; import { findEventsInCache } from "../../services/EventCache/findEventInCache"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; @@ -24,7 +24,16 @@ export const removeEventAttendee: MutationResolvers["removeEventAttendee"] = USER_NOT_FOUND_ERROR.PARAM ); } - + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let currentEvent: InterfaceEvent | null; const eventFoundInCache = await findEventsInCache([args.data.eventId]); @@ -53,7 +62,7 @@ export const removeEventAttendee: MutationResolvers["removeEventAttendee"] = (admin) => admin.toString() === context.userId.toString() ); - if (!isUserEventAdmin && currentUser.userType !== "SUPERADMIN") { + if (!isUserEventAdmin && currentUserAppProfile.isSuperAdmin === false) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, diff --git a/src/resolvers/Mutation/removeGroupChat.ts b/src/resolvers/Mutation/removeGroupChat.ts index b2e9d6d34f..c5e2998e62 100644 --- a/src/resolvers/Mutation/removeGroupChat.ts +++ b/src/resolvers/Mutation/removeGroupChat.ts @@ -49,8 +49,7 @@ export const removeGroupChat: MutationResolvers["removeGroupChat"] = async ( organization = await Organization.findOne({ _id: groupChat.organization, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks if an organization with _id === groupChat.organization exists. diff --git a/src/resolvers/Mutation/removeMember.ts b/src/resolvers/Mutation/removeMember.ts index 56e394af54..d71316db0e 100644 --- a/src/resolvers/Mutation/removeMember.ts +++ b/src/resolvers/Mutation/removeMember.ts @@ -1,19 +1,19 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import type { InterfaceOrganization } from "../../models"; -import { User, Organization } from "../../models"; -import { adminCheck } from "../../utilities"; +import { Types } from "mongoose"; import { + ADMIN_REMOVING_ADMIN, + ADMIN_REMOVING_CREATOR, + MEMBER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_FOUND_ERROR, - MEMBER_NOT_FOUND_ERROR, USER_REMOVING_SELF, - ADMIN_REMOVING_ADMIN, - ADMIN_REMOVING_CREATOR, } from "../../constants"; -import { Types } from "mongoose"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceOrganization } from "../../models"; +import { Organization, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { adminCheck } from "../../utilities"; /** * This function enables to remove a member. * @param _parent - parent of current request @@ -41,8 +41,7 @@ export const removeMember: MutationResolvers["removeMember"] = async ( organization = await Organization.findOne({ _id: args.data.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } else { organization = organizationFoundInCache[0]; } @@ -144,8 +143,7 @@ export const removeMember: MutationResolvers["removeMember"] = async ( new: true, } ).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); // Remove organization's id from joinedOrganizations list on user. await User.updateOne( diff --git a/src/resolvers/Mutation/removeOrganization.ts b/src/resolvers/Mutation/removeOrganization.ts index 5396eaaef8..e57050f010 100644 --- a/src/resolvers/Mutation/removeOrganization.ts +++ b/src/resolvers/Mutation/removeOrganization.ts @@ -1,22 +1,31 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../constants"; import { errors, requestContext } from "../../libraries"; +import type { + InterfaceAppUserProfile, + InterfaceEvent, + InterfaceOrganization, + InterfaceUser, +} from "../../models"; import { - User, - Organization, - Post, + ActionItem, + ActionItemCategory, + AppUserProfile, Comment, MembershipRequest, - ActionItemCategory, - ActionItem, + Organization, + Post, + User, } from "../../models"; -import { superAdminCheck } from "../../utilities"; -import { - USER_NOT_FOUND_ERROR, - ORGANIZATION_NOT_FOUND_ERROR, -} from "../../constants"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; import { deleteOrganizationFromCache } from "../../services/OrganizationCache/deleteOrganizationFromCache"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; + +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { superAdminCheck } from "../../utilities"; import { deletePreviousImage as deleteImage } from "../../utilities/encodedImageStorage/deletePreviousImage"; /** * This function enables to remove an organization. @@ -27,6 +36,7 @@ import { deletePreviousImage as deleteImage } from "../../utilities/encodedImage * 1. If the user exists. * 2. If the organization exists * 3. If the user is the creator of the organization. + * 4. If the user has appUserProfile. * @returns Updated user. */ export const removeOrganization: MutationResolvers["removeOrganization"] = @@ -43,6 +53,17 @@ export const removeOrganization: MutationResolvers["removeOrganization"] = USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let organization; @@ -54,8 +75,9 @@ export const removeOrganization: MutationResolvers["removeOrganization"] = organization = await Organization.findOne({ _id: args.id, }).lean(); - - await cacheOrganizations([organization!]); + if (organization != null) { + await cacheOrganizations([organization]); + } } // Checks whether organization exists. @@ -67,16 +89,16 @@ export const removeOrganization: MutationResolvers["removeOrganization"] = ); } // Checks whether currentUser is a SUPERADMIN - superAdminCheck(currentUser); + superAdminCheck(currentUserAppProfile); // Remove each post and comments associated to it for organization.posts list. await Post.deleteMany({ _id: { $in: organization.posts } }); await Comment.deleteMany({ postId: { $in: organization.posts } }); - // Remove organization._id from createdOrganizations list of currentUser. - await User.updateOne( + // Remove organization._id from createdOrganizations list of currentUserAppProfile*. + await AppUserProfile.updateOne( { - _id: currentUser._id, + _id: currentUserAppProfile._id, }, { $pull: { @@ -155,11 +177,31 @@ export const removeOrganization: MutationResolvers["removeOrganization"] = if (organization?.image) { await deleteImage(organization?.image); } - - // Returns updated currentUser. - return await User.findOne({ + const updatedUser: InterfaceUser = await User.findOne({ _id: currentUser._id, }) .select(["-password"]) .lean(); + const updatedAppUserProfile: InterfaceAppUserProfile = + await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + + // Returns updated currentUser. + return { + user: updatedUser, + appUserProfile: { + _id: updatedAppUserProfile._id.toString(), + userId: updatedAppUserProfile.userId as InterfaceUser, + adminFor: updatedAppUserProfile.adminFor as InterfaceOrganization[], + appLanguageCode: updatedAppUserProfile.appLanguageCode, + isSuperAdmin: updatedAppUserProfile.isSuperAdmin, + pluginCreationAllowed: updatedAppUserProfile.pluginCreationAllowed, + tokenVersion: updatedAppUserProfile.tokenVersion, + eventAdmin: updatedAppUserProfile.eventAdmin as InterfaceEvent[], + createdEvents: updatedAppUserProfile.createdEvents as InterfaceEvent[], + createdOrganizations: + updatedAppUserProfile.createdOrganizations as InterfaceOrganization[], + }, + }; }; diff --git a/src/resolvers/Mutation/removeOrganizationCustomField.ts b/src/resolvers/Mutation/removeOrganizationCustomField.ts index 5009c2fcf6..dce6295fd8 100644 --- a/src/resolvers/Mutation/removeOrganizationCustomField.ts +++ b/src/resolvers/Mutation/removeOrganizationCustomField.ts @@ -1,3 +1,4 @@ +import { Types } from "mongoose"; import { CUSTOM_FIELD_NOT_FOUND, ORGANIZATION_NOT_FOUND_ERROR, @@ -5,7 +6,12 @@ import { USER_NOT_FOUND_ERROR, } from "../../constants"; import { errors, requestContext } from "../../libraries"; -import { OrganizationCustomField, Organization, User } from "../../models"; +import { + AppUserProfile, + Organization, + OrganizationCustomField, + User, +} from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** @@ -18,6 +24,7 @@ import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; * 2. If the organization exists. * 3. If the user is an admin for the organization. * 4. If the custom field to be removed exists + * 5. If the user has appUserProfile * @returns Deleted Organization Custom Field. */ @@ -36,6 +43,16 @@ export const removeOrganizationCustomField: MutationResolvers["removeOrganizatio USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const organization = await Organization.findOne({ _id: organizationId, @@ -49,12 +66,13 @@ export const removeOrganizationCustomField: MutationResolvers["removeOrganizatio ); } - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(organization._id) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (orgId) => + orgId && Types.ObjectId(orgId.toString()).equals(organization._id) ); if ( - !(currentUserIsOrganizationAdmin || currentUser.userType === "SUPERADMIN") + !(currentUserIsOrganizationAdmin || currentUserAppProfile.isSuperAdmin) ) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), diff --git a/src/resolvers/Mutation/removeOrganizationImage.ts b/src/resolvers/Mutation/removeOrganizationImage.ts index 876069d2b9..76f84213c3 100644 --- a/src/resolvers/Mutation/removeOrganizationImage.ts +++ b/src/resolvers/Mutation/removeOrganizationImage.ts @@ -1,13 +1,14 @@ import { - ORGANIZATION_NOT_FOUND_ERROR, ORGANIZATION_IMAGE_NOT_FOUND_ERROR, + ORGANIZATION_NOT_FOUND_ERROR, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; +import type { InterfaceOrganization } from "../../models"; import { Organization } from "../../models"; -import { adminCheck, deleteImage } from "../../utilities"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { adminCheck, deleteImage } from "../../utilities"; /** * This function enables to remove an organization's image. * @param _parent - parent of current request @@ -33,8 +34,7 @@ export const removeOrganizationImage: MutationResolvers["removeOrganizationImage organization = await Organization.findOne({ _id: args.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks whether organization exists. @@ -79,5 +79,5 @@ export const removeOrganizationImage: MutationResolvers["removeOrganizationImage await cacheOrganizations([updatedOrganization]); } - return updatedOrganization!; + return updatedOrganization as InterfaceOrganization; }; diff --git a/src/resolvers/Mutation/removePost.ts b/src/resolvers/Mutation/removePost.ts index fb4077cd5f..12c0bc8b38 100644 --- a/src/resolvers/Mutation/removePost.ts +++ b/src/resolvers/Mutation/removePost.ts @@ -1,16 +1,17 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import type { InterfacePost } from "../../models"; -import { User, Post, Organization } from "../../models"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, POST_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import type { InterfacePost } from "../../models"; +import { AppUserProfile, Organization, Post, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { cachePosts } from "../../services/PostCache/cachePosts"; import { deletePostFromCache } from "../../services/PostCache/deletePostFromCache"; import { findPostsInCache } from "../../services/PostCache/findPostsInCache"; -import { cachePosts } from "../../services/PostCache/cachePosts"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { deletePreviousImage as deleteImage } from "../../utilities/encodedImageStorage/deletePreviousImage"; import { deletePreviousVideo as deleteVideo } from "../../utilities/encodedVideoStorage/deletePreviousVideo"; /** @@ -23,6 +24,7 @@ import { deletePreviousVideo as deleteVideo } from "../../utilities/encodedVideo * 2. If the post exists * 3. If the user is the creator of the post. * 4. If the user to be removed is a member of the organization. + * 5. If the user has appUserProfile. * @returns Deleted Post. */ export const removePost: MutationResolvers["removePost"] = async ( @@ -43,6 +45,16 @@ export const removePost: MutationResolvers["removePost"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let post: InterfacePost | null; @@ -70,9 +82,10 @@ export const removePost: MutationResolvers["removePost"] = async ( // Checks whether currentUser is allowed to delete the post or not. const isCreator = post.creatorId.equals(context.userId); - const isSuperAdmin = currentUser?.userType === "SUPERADMIN"; - const isAdminOfPostOrganization = currentUser?.adminFor.some((orgID) => - orgID.equals(post?.organization) + const isSuperAdmin = currentUserAppProfile.isSuperAdmin; + const isAdminOfPostOrganization = currentUserAppProfile?.adminFor.some( + (orgID) => + orgID && Types.ObjectId(orgID?.toString()).equals(post?.organization) ); if (!isCreator && !isSuperAdmin && !isAdminOfPostOrganization) { diff --git a/src/resolvers/Mutation/removeSampleOrganization.ts b/src/resolvers/Mutation/removeSampleOrganization.ts index e1bf9f7667..efaeb75c3f 100644 --- a/src/resolvers/Mutation/removeSampleOrganization.ts +++ b/src/resolvers/Mutation/removeSampleOrganization.ts @@ -1,12 +1,13 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { removeSampleOrganization as removeSampleOrgUtil } from "../../utilities/removeSampleOrganizationUtil"; -import { SampleData, User } from "../../models"; -import { errors, requestContext } from "../../libraries"; +import { Types } from "mongoose"; import { ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, SampleData, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { removeSampleOrganization as removeSampleOrgUtil } from "../../utilities/removeSampleOrganizationUtil"; export const removeSampleOrganization: MutationResolvers["removeSampleOrganization"] = async (_parent, _args, _context) => { @@ -21,13 +22,10 @@ export const removeSampleOrganization: MutationResolvers["removeSampleOrganizati USER_NOT_FOUND_ERROR.PARAM ); } - - if ( - !( - currentUser.userType === "SUPERADMIN" || - currentUser.userType === "ADMIN" - ) - ) { + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, @@ -47,6 +45,20 @@ export const removeSampleOrganization: MutationResolvers["removeSampleOrganizati ); } + const currentUserOrgAdmin = currentUserAppProfile.adminFor.some( + (org) => + org && + Types.ObjectId(org.toString()).equals(existingOrganization.documentId) + ); + + if (!currentUserAppProfile.isSuperAdmin && !currentUserOrgAdmin) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } + await removeSampleOrgUtil(); return true; }; diff --git a/src/resolvers/Mutation/removeUserCustomData.ts b/src/resolvers/Mutation/removeUserCustomData.ts index c1ec2228ed..324f4137bf 100644 --- a/src/resolvers/Mutation/removeUserCustomData.ts +++ b/src/resolvers/Mutation/removeUserCustomData.ts @@ -1,13 +1,14 @@ -import { UserCustomData } from "../../models/UserCustomData"; -import { Organization, User } from "../../models"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; +import { Types } from "mongoose"; import { + CUSTOM_DATA_NOT_FOUND, ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, - CUSTOM_DATA_NOT_FOUND, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, Organization, User } from "../../models"; +import { UserCustomData } from "../../models/UserCustomData"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const removeUserCustomData: MutationResolvers["removeUserCustomData"] = async (_parent, args, context) => { @@ -24,7 +25,16 @@ export const removeUserCustomData: MutationResolvers["removeUserCustomData"] = USER_NOT_FOUND_ERROR.PARAM ); } - + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const organization = await Organization.findOne({ _id: organizationId, }).lean(); @@ -37,12 +47,13 @@ export const removeUserCustomData: MutationResolvers["removeUserCustomData"] = ); } - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(organization._id) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (orgId) => + orgId && Types.ObjectId(orgId?.toString()).equals(organization._id) ); if ( - !(currentUserIsOrganizationAdmin || currentUser.userType === "SUPERADMIN") + !(currentUserIsOrganizationAdmin || currentUserAppProfile.isSuperAdmin) ) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), diff --git a/src/resolvers/Mutation/removeUserFamily.ts b/src/resolvers/Mutation/removeUserFamily.ts index 38181ad1a9..f8329e6023 100644 --- a/src/resolvers/Mutation/removeUserFamily.ts +++ b/src/resolvers/Mutation/removeUserFamily.ts @@ -2,10 +2,12 @@ import { USER_FAMILY_NOT_FOUND_ERROR, USER_NOT_FOUND_ERROR, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import { UserFamily } from "../../models/userFamily"; -import { User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; + +import { AppUserProfile, User } from "../../models"; + import { superAdminCheck } from "../../utilities"; /** * This function enables to remove a user family. @@ -39,8 +41,18 @@ export const removeUserFamily: MutationResolvers["removeUserFamily"] = async ( ); } - // Checks whether currentUser is a SUPERADMIN - superAdminCheck(currentUser); + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.NotFoundError( + requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); + } + // Check whether the user is super admin. + superAdminCheck(currentUserAppProfile); // Checks if a family with _id === args.familyId exists if (!userFamily) { diff --git a/src/resolvers/Mutation/removeUserFromGroupChat.ts b/src/resolvers/Mutation/removeUserFromGroupChat.ts index 55615e0281..1716408406 100644 --- a/src/resolvers/Mutation/removeUserFromGroupChat.ts +++ b/src/resolvers/Mutation/removeUserFromGroupChat.ts @@ -48,8 +48,7 @@ export const removeUserFromGroupChat: MutationResolvers["removeUserFromGroupChat organization = await Organization.findOne({ _id: groupChat.organization, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks whether organization exists. diff --git a/src/resolvers/Mutation/removeUserTag.ts b/src/resolvers/Mutation/removeUserTag.ts index ce0c84e5eb..e989d7642e 100644 --- a/src/resolvers/Mutation/removeUserTag.ts +++ b/src/resolvers/Mutation/removeUserTag.ts @@ -1,11 +1,17 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { User, OrganizationTagUser, TagUser } from "../../models"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, TAG_NOT_FOUND, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { + AppUserProfile, + OrganizationTagUser, + TagUser, + User, +} from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const removeUserTag: MutationResolvers["removeUserTag"] = async ( _parent, @@ -24,6 +30,16 @@ export const removeUserTag: MutationResolvers["removeUserTag"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Get the tag object const tag = await OrganizationTagUser.findOne({ @@ -39,15 +55,14 @@ export const removeUserTag: MutationResolvers["removeUserTag"] = async ( } // Boolean to determine whether user is an admin of organization of the tag - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(tag.organizationId) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (organization) => + organization && + Types.ObjectId(organization.toString()).equals(tag.organizationId) ); // Checks whether currentUser cannot delete the tag folder. - if ( - !(currentUser.userType === "SUPERADMIN") && - !currentUserIsOrganizationAdmin - ) { + if (!currentUserAppProfile.isSuperAdmin && !currentUserIsOrganizationAdmin) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, diff --git a/src/resolvers/Mutation/revokeRefreshTokenForUser.ts b/src/resolvers/Mutation/revokeRefreshTokenForUser.ts index a2dadd821d..6597776f93 100644 --- a/src/resolvers/Mutation/revokeRefreshTokenForUser.ts +++ b/src/resolvers/Mutation/revokeRefreshTokenForUser.ts @@ -1,5 +1,5 @@ +import { AppUserProfile } from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; /** * This function creates a refresh token for user. * @param _parent - parent of current request @@ -8,9 +8,9 @@ import { User } from "../../models"; */ export const revokeRefreshTokenForUser: MutationResolvers["revokeRefreshTokenForUser"] = async (_parent, args, context) => { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: context.userId, + userId: context.userId, }, { $unset: { token: 1 }, diff --git a/src/resolvers/Mutation/saveFcmToken.ts b/src/resolvers/Mutation/saveFcmToken.ts index 8988d3d603..b7b1161134 100644 --- a/src/resolvers/Mutation/saveFcmToken.ts +++ b/src/resolvers/Mutation/saveFcmToken.ts @@ -1,5 +1,5 @@ +import { AppUserProfile } from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; /** * This function enables to save Fcm Token. * @param _parent - parent of current request @@ -14,9 +14,9 @@ export const saveFcmToken: MutationResolvers["saveFcmToken"] = async ( args, context ) => { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: context.userId, + userId: context.userId, }, { $set: { diff --git a/src/resolvers/Mutation/signUp.ts b/src/resolvers/Mutation/signUp.ts index 1ef2131566..e98dadd402 100644 --- a/src/resolvers/Mutation/signUp.ts +++ b/src/resolvers/Mutation/signUp.ts @@ -1,22 +1,26 @@ import bcrypt from "bcryptjs"; import { + EMAIL_ALREADY_EXISTS_ERROR, LAST_RESORT_SUPERADMIN_EMAIL, //LENGTH_VALIDATION_ERROR, ORGANIZATION_NOT_FOUND_ERROR, - EMAIL_ALREADY_EXISTS_ERROR, - //REGEX_VALIDATION_ERROR, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; -import { User, Organization } from "../../models"; +import type { + InterfaceEvent, + InterfaceOrganization, + InterfaceUser, +} from "../../models"; +import { AppUserProfile, Organization, User } from "../../models"; +import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; +import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { + copyToClipboard, createAccessToken, createRefreshToken, - copyToClipboard, } from "../../utilities"; import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; -import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; -import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; //import { isValidString } from "../../libraries/validators/validateString"; //import { validatePassword } from "../../libraries/validators/validatePassword"; /** @@ -52,7 +56,7 @@ export const signUp: MutationResolvers["signUp"] = async (_parent, args) => { _id: args.data.organizationUserBelongsToId, }).lean(); - await cacheOrganizations([organization!]); + await cacheOrganizations([organization as InterfaceOrganization]); } if (!organization) { @@ -120,29 +124,64 @@ export const signUp: MutationResolvers["signUp"] = async (_parent, args) => { const isLastResortSuperAdmin = args.data.email === LAST_RESORT_SUPERADMIN_EMAIL; - const createdUser = await User.create({ + let createdUser = await User.create({ ...args.data, email: args.data.email.toLowerCase(), // ensure all emails are stored as lowercase to prevent duplicated due to comparison errors image: uploadImageFileName ? uploadImageFileName : null, password: hashedPassword, - userType: isLastResortSuperAdmin ? "SUPERADMIN" : "USER", + // userType: isLastResortSuperAdmin ? "SUPERADMIN" : "USER", adminApproved: isLastResortSuperAdmin, }); + const appUserProfile = await AppUserProfile.create({ + userId: createdUser._id, + appLanguageCode: args.data.appLanguageCode || "en", + isSuperAdmin: isLastResortSuperAdmin, + }); + + const updatedUser = await User.findOneAndUpdate( + { + _id: createdUser._id, + }, + { + appUserProfileId: appUserProfile._id, + }, + { + new: true, + } + ); + + if (updatedUser) { + createdUser = updatedUser; + } else { + throw new Error("Failed to update user."); + } - const accessToken = await createAccessToken(createdUser); - const refreshToken = await createRefreshToken(createdUser); + const accessToken = await createAccessToken(createdUser, appUserProfile); + const refreshToken = await createRefreshToken(createdUser, appUserProfile); copyToClipboard(`{ "Authorization": "Bearer ${accessToken}" }`); - const filteredCreatedUser = createdUser.toObject(); + const filteredCreatedUser = updatedUser.toObject(); - // @ts-ignore delete filteredCreatedUser.password; return { user: filteredCreatedUser, + appUserProfile: { + _id: appUserProfile._id.toString(), + userId: appUserProfile.userId as InterfaceUser, + adminFor: appUserProfile.adminFor as InterfaceOrganization[], + appLanguageCode: appUserProfile.appLanguageCode, + isSuperAdmin: appUserProfile.isSuperAdmin, + pluginCreationAllowed: appUserProfile.pluginCreationAllowed, + tokenVersion: appUserProfile.tokenVersion, + eventAdmin: appUserProfile.eventAdmin as InterfaceEvent[], + createdEvents: appUserProfile.createdEvents as InterfaceEvent[], + createdOrganizations: + appUserProfile.createdOrganizations as InterfaceOrganization[], + }, accessToken, refreshToken, }; diff --git a/src/resolvers/Mutation/togglePostPin.ts b/src/resolvers/Mutation/togglePostPin.ts index ebd55e4ee5..cedb9c10db 100644 --- a/src/resolvers/Mutation/togglePostPin.ts +++ b/src/resolvers/Mutation/togglePostPin.ts @@ -1,20 +1,21 @@ +import { Types } from "mongoose"; import { + LENGTH_VALIDATION_ERROR, + PLEASE_PROVIDE_TITLE, POST_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_AUTHORIZED_TO_PIN, - PLEASE_PROVIDE_TITLE, - LENGTH_VALIDATION_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; +import { isValidString } from "../../libraries/validators/validateString"; import type { InterfacePost } from "../../models"; -import { User, Post, Organization } from "../../models"; +import { AppUserProfile, Organization, Post, User } from "../../models"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; -import { Types } from "mongoose"; -import { findPostsInCache } from "../../services/PostCache/findPostsInCache"; import { cachePosts } from "../../services/PostCache/cachePosts"; -import { isValidString } from "../../libraries/validators/validateString"; +import { findPostsInCache } from "../../services/PostCache/findPostsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const togglePostPin: MutationResolvers["togglePostPin"] = async ( _parent, @@ -34,7 +35,16 @@ export const togglePostPin: MutationResolvers["togglePostPin"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } - + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Check if the post object exists let post: InterfacePost | null; @@ -60,8 +70,10 @@ export const togglePostPin: MutationResolvers["togglePostPin"] = async ( } // Check if the current user is authorized to perform the operation - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organizationId) => organizationId.equals(post?.organization) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (organizationId) => + organizationId && + Types.ObjectId(organizationId.toString()).equals(post?.organization) ); if ( @@ -88,8 +100,9 @@ export const togglePostPin: MutationResolvers["togglePostPin"] = async ( organization = await Organization.findOne({ _id: post.organization, }).lean(); - - await cacheOrganizations([organization!]); + if (organization !== null) { + await cacheOrganizations([organization]); + } } const currentPostIsPinned = organization?.pinnedPosts.some((postID) => Types.ObjectId(postID).equals(args.id) @@ -130,7 +143,7 @@ export const togglePostPin: MutationResolvers["togglePostPin"] = async ( await cachePosts([updatedPost]); } - return updatedPost!; + return updatedPost as InterfacePost; } else { if (!args.title) { throw new errors.InputValidationError( @@ -184,6 +197,6 @@ export const togglePostPin: MutationResolvers["togglePostPin"] = async ( await cachePosts([updatedPost]); } - return updatedPost!; + return updatedPost as InterfacePost; } }; diff --git a/src/resolvers/Mutation/unassignUserTag.ts b/src/resolvers/Mutation/unassignUserTag.ts index b1fe4b1897..2acabfdbf7 100644 --- a/src/resolvers/Mutation/unassignUserTag.ts +++ b/src/resolvers/Mutation/unassignUserTag.ts @@ -1,12 +1,18 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { User, OrganizationTagUser, TagUser } from "../../models"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, TAG_NOT_FOUND, USER_DOES_NOT_HAVE_THE_TAG, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { + AppUserProfile, + OrganizationTagUser, + TagUser, + User, +} from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const unassignUserTag: MutationResolvers["unassignUserTag"] = async ( _parent, @@ -25,6 +31,16 @@ export const unassignUserTag: MutationResolvers["unassignUserTag"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Get the tag object const tag = await OrganizationTagUser.findOne({ @@ -40,15 +56,14 @@ export const unassignUserTag: MutationResolvers["unassignUserTag"] = async ( } // Boolean to determine whether user is an admin of organization of the tag. - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(tag?.organizationId) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (organization) => + organization && + Types.ObjectId(organization.toString()).equals(tag?.organizationId) ); // Checks whether currentUser can assign the tag or not. - if ( - !currentUserIsOrganizationAdmin && - !(currentUser.userType === "SUPERADMIN") - ) { + if (!currentUserIsOrganizationAdmin && !currentUserAppProfile.isSuperAdmin) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, diff --git a/src/resolvers/Mutation/unblockUser.ts b/src/resolvers/Mutation/unblockUser.ts index 7b7087e053..ee25c0dd06 100644 --- a/src/resolvers/Mutation/unblockUser.ts +++ b/src/resolvers/Mutation/unblockUser.ts @@ -37,8 +37,7 @@ export const unblockUser: MutationResolvers["unblockUser"] = async ( organization = await Organization.findOne({ _id: args.organizationId, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } else { organization = organizationFoundInCache[0]; } diff --git a/src/resolvers/Mutation/unlikePost.ts b/src/resolvers/Mutation/unlikePost.ts index 2f36334b7a..34df3aab9b 100644 --- a/src/resolvers/Mutation/unlikePost.ts +++ b/src/resolvers/Mutation/unlikePost.ts @@ -1,10 +1,10 @@ import { POST_NOT_FOUND_ERROR } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import type { InterfacePost } from "../../models"; import { Post } from "../../models"; -import { findPostsInCache } from "../../services/PostCache/findPostsInCache"; import { cachePosts } from "../../services/PostCache/cachePosts"; +import { findPostsInCache } from "../../services/PostCache/findPostsInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to unlike a post. * @param _parent - parent of current request @@ -69,7 +69,7 @@ export const unlikePost: MutationResolvers["unlikePost"] = async ( await cachePosts([updatedPost]); } - return updatedPost!; + return updatedPost; } return post; diff --git a/src/resolvers/Mutation/unregisterForEventByUser.ts b/src/resolvers/Mutation/unregisterForEventByUser.ts index a7ef40cce1..b79e40ba2d 100644 --- a/src/resolvers/Mutation/unregisterForEventByUser.ts +++ b/src/resolvers/Mutation/unregisterForEventByUser.ts @@ -1,13 +1,13 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import type { InterfaceEvent } from "../../models"; -import { Event, EventAttendee } from "../../models"; import { EVENT_NOT_FOUND_ERROR, USER_ALREADY_UNREGISTERED_ERROR, } from "../../constants"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceEvent } from "../../models"; +import { Event, EventAttendee } from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables a user to unregister from an event. diff --git a/src/resolvers/Mutation/updateActionItem.ts b/src/resolvers/Mutation/updateActionItem.ts index 7a17387f2b..e30ca66d51 100644 --- a/src/resolvers/Mutation/updateActionItem.ts +++ b/src/resolvers/Mutation/updateActionItem.ts @@ -1,3 +1,4 @@ +import { Types } from "mongoose"; import { ACTION_ITEM_NOT_FOUND_ERROR, EVENT_NOT_FOUND_ERROR, @@ -5,13 +6,12 @@ import { USER_NOT_FOUND_ERROR, USER_NOT_MEMBER_FOR_ORGANIZATION, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import type { InterfaceEvent } from "../../models"; -import { User, ActionItem, Event } from "../../models"; -import { Types } from "mongoose"; -import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import { ActionItem, AppUserProfile, Event, User } from "../../models"; import { cacheEvents } from "../../services/EventCache/cacheEvents"; +import { findEventsInCache } from "../../services/EventCache/findEventInCache"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to update an action item. * @param _parent - parent of current request @@ -23,6 +23,7 @@ import { cacheEvents } from "../../services/EventCache/cacheEvents"; * 2. If the action item exists. * 4. If the new asignee is a member of the organization. * 5. If the user is authorized. + * 6. If the user has appUserProfile. * @returns Updated action item. */ @@ -40,7 +41,7 @@ export const updateActionItem: MutationResolvers["updateActionItem"] = async ( args, context ) => { - const currentUser = await User.findOne({ + const currentUser = await User.findById({ _id: context.userId, }); @@ -52,6 +53,16 @@ export const updateActionItem: MutationResolvers["updateActionItem"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const actionItem = await ActionItem.findOne({ _id: args.id, @@ -108,10 +119,10 @@ export const updateActionItem: MutationResolvers["updateActionItem"] = async ( } } - const currentUserIsOrgAdmin = currentUser.adminFor.some( + const currentUserIsOrgAdmin = currentUserAppProfile.adminFor.some( (ogranizationId) => ogranizationId === actionItem.actionItemCategoryId.organizationId || - Types.ObjectId(ogranizationId).equals( + Types.ObjectId(ogranizationId?.toString()).equals( actionItem.actionItemCategoryId.organizationId ) ); @@ -155,7 +166,7 @@ export const updateActionItem: MutationResolvers["updateActionItem"] = async ( if ( currentUserIsEventAdmin === false && currentUserIsOrgAdmin === false && - currentUser.userType !== "SUPERADMIN" + currentUserAppProfile.isSuperAdmin === false ) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), diff --git a/src/resolvers/Mutation/updateEvent.ts b/src/resolvers/Mutation/updateEvent.ts index a33a38a740..75d2c7d7a4 100644 --- a/src/resolvers/Mutation/updateEvent.ts +++ b/src/resolvers/Mutation/updateEvent.ts @@ -1,7 +1,7 @@ import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; import type { InterfaceEvent } from "../../models"; -import { User, Event } from "../../models"; +import { User, Event, AppUserProfile } from "../../models"; import { USER_NOT_FOUND_ERROR, EVENT_NOT_FOUND_ERROR, @@ -40,6 +40,16 @@ export const updateEvent: MutationResolvers["updateEvent"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } let event: InterfaceEvent | null; @@ -74,7 +84,7 @@ export const updateEvent: MutationResolvers["updateEvent"] = async ( // checks if current user is an admin of the event with _id === args.id if ( currentUserIsEventAdmin === false && - currentUser.userType !== "SUPERADMIN" + currentUserAppProfile.isSuperAdmin === false ) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), @@ -120,6 +130,7 @@ export const updateEvent: MutationResolvers["updateEvent"] = async ( _id: args.id, }, { + // eslint-disable-next-line @typescript-eslint/no-explicit-any ...(args.data as any), }, { @@ -131,5 +142,5 @@ export const updateEvent: MutationResolvers["updateEvent"] = async ( await cacheEvents([updatedEvent]); } - return updatedEvent!; + return updatedEvent as InterfaceEvent; }; diff --git a/src/resolvers/Mutation/updateOrganization.ts b/src/resolvers/Mutation/updateOrganization.ts index 996d050574..7976d69afc 100644 --- a/src/resolvers/Mutation/updateOrganization.ts +++ b/src/resolvers/Mutation/updateOrganization.ts @@ -1,12 +1,13 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; +import { ORGANIZATION_NOT_FOUND_ERROR } from "../../constants"; import { errors, requestContext } from "../../libraries"; +import type { InterfaceOrganization } from "../../models"; import { Organization } from "../../models"; -import { ORGANIZATION_NOT_FOUND_ERROR } from "../../constants"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { adminCheck } from "../../utilities"; -import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; import { cacheOrganizations } from "../../services/OrganizationCache/cacheOrganizations"; import { findOrganizationsInCache } from "../../services/OrganizationCache/findOrganizationsInCache"; +import { uploadEncodedImage } from "../../utilities/encodedImageStorage/uploadEncodedImage"; /** * This function enables to update an organization. * @param _parent - parent of current request @@ -30,8 +31,7 @@ export const updateOrganization: MutationResolvers["updateOrganization"] = organization = await Organization.findOne({ _id: args.id, }).lean(); - - await cacheOrganizations([organization!]); + if (organization) await cacheOrganizations([organization]); } // Checks if organization with _id === args.id exists. @@ -73,5 +73,5 @@ export const updateOrganization: MutationResolvers["updateOrganization"] = await cacheOrganizations([updatedOrganization]); } - return updatedOrganization!; + return updatedOrganization as InterfaceOrganization; }; diff --git a/src/resolvers/Mutation/updatePluginStatus.ts b/src/resolvers/Mutation/updatePluginStatus.ts index b819062d30..d5952adbfc 100644 --- a/src/resolvers/Mutation/updatePluginStatus.ts +++ b/src/resolvers/Mutation/updatePluginStatus.ts @@ -1,8 +1,9 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { Plugin } from "../../models"; import mongoose from "mongoose"; -import { errors, requestContext } from "../../libraries"; import { PLUGIN_NOT_FOUND } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import type { InterfacePlugin } from "../../models"; +import { Plugin } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** * This function enables to update plugin install status. @@ -11,10 +12,8 @@ import { PLUGIN_NOT_FOUND } from "../../constants"; * @param _context - context of entire application * @returns Updated PLugin status. */ - -// @ts-ignore export const updatePluginStatus: MutationResolvers["updatePluginStatus"] = - async (_parent, args, context) => { + async (_parent, args, context): Promise => { const uid = args.id; // const currOrgID = mongoose.Types.ObjectId(args.orgId) ; const currOrgID = args.orgId; @@ -30,11 +29,11 @@ export const updatePluginStatus: MutationResolvers["updatePluginStatus"] = } let uninstalledOrgsList = plugin.uninstalledOrgs; - // @ts-ignore - if (uninstalledOrgsList.includes(currOrgID)) { + + if (uninstalledOrgsList.includes(mongoose.Types.ObjectId(currOrgID))) { //if already uninstalled then install it by removing from array uninstalledOrgsList = uninstalledOrgsList.filter( - (oid: any) => oid != currOrgID + (oid: unknown) => oid != currOrgID ); } else { //not already present then uninstall plugin on that org by adding it to the list @@ -58,5 +57,5 @@ export const updatePluginStatus: MutationResolvers["updatePluginStatus"] = context.pubsub.publish("TALAWA_PLUGIN_UPDATED", { onPluginUpdate: res, }); - return res; + return res as InterfacePlugin; }; diff --git a/src/resolvers/Mutation/updateUserPassword.ts b/src/resolvers/Mutation/updateUserPassword.ts index ff20e208c4..b9d27ce165 100644 --- a/src/resolvers/Mutation/updateUserPassword.ts +++ b/src/resolvers/Mutation/updateUserPassword.ts @@ -1,11 +1,18 @@ +import bcrypt from "bcryptjs"; import { INVALID_CREDENTIALS_ERROR, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../constants"; -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; import { errors, requestContext } from "../../libraries"; -import { User } from "../../models"; -import bcrypt from "bcryptjs"; +import type { + InterfaceAppUserProfile, + InterfaceEvent, + InterfaceOrganization, + InterfaceUser, +} from "../../models"; +import { AppUserProfile, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const updateUserPassword: MutationResolvers["updateUserPassword"] = async (_parent, args, context) => { @@ -20,10 +27,19 @@ export const updateUserPassword: MutationResolvers["updateUserPassword"] = USER_NOT_FOUND_ERROR.PARAM ); } - + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + USER_NOT_AUTHORIZED_ERROR.MESSAGE, + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } const isPasswordValid = await bcrypt.compare( args.data.previousPassword, - currentUser.password + currentUser?.password || "" ); // Checks whether password is invalid. @@ -58,19 +74,48 @@ export const updateUserPassword: MutationResolvers["updateUserPassword"] = } const hashedPassword = await bcrypt.hash(args.data.newPassword, 12); - - return await User.findOneAndUpdate( + const updatedUser = await User.findOneAndUpdate( { _id: context.userId, }, { $set: { password: hashedPassword, - token: null, }, }, { new: true, } - ).lean(); + ); + const updatedAppUserProfile: InterfaceAppUserProfile = + await AppUserProfile.findOneAndUpdate( + { + userId: context.userId, + }, + { + $set: { + token: null, + }, + }, + { + new: true, + } + ).lean(); + + return { + user: updatedUser as InterfaceUser, + appUserProfile: { + _id: updatedAppUserProfile._id.toString(), + userId: updatedAppUserProfile.userId as InterfaceUser, + adminFor: updatedAppUserProfile.adminFor as InterfaceOrganization[], + appLanguageCode: updatedAppUserProfile.appLanguageCode, + isSuperAdmin: updatedAppUserProfile.isSuperAdmin, + pluginCreationAllowed: updatedAppUserProfile.pluginCreationAllowed, + tokenVersion: updatedAppUserProfile.tokenVersion, + eventAdmin: updatedAppUserProfile.eventAdmin as InterfaceEvent[], + createdEvents: updatedAppUserProfile.createdEvents as InterfaceEvent[], + createdOrganizations: + updatedAppUserProfile.createdOrganizations as InterfaceOrganization[], + }, + }; }; diff --git a/src/resolvers/Mutation/updateUserProfile.ts b/src/resolvers/Mutation/updateUserProfile.ts index d63d6821a5..b7fa960511 100644 --- a/src/resolvers/Mutation/updateUserProfile.ts +++ b/src/resolvers/Mutation/updateUserProfile.ts @@ -128,9 +128,11 @@ export const updateUserProfile: MutationResolvers["updateUserProfile"] = async ( runValidators: true, } ).lean(); - updatedUser!.image = updatedUser?.image - ? `${context.apiRootUrl}${updatedUser?.image}` - : null; + if (updatedUser) { + updatedUser.image = updatedUser?.image + ? `${context.apiRootUrl}${updatedUser?.image}` + : null; + } return updatedUser ?? ({} as InterfaceUser); }; diff --git a/src/resolvers/Mutation/updateUserRoleInOrganization.ts b/src/resolvers/Mutation/updateUserRoleInOrganization.ts index 44ab448d76..109aeb5138 100644 --- a/src/resolvers/Mutation/updateUserRoleInOrganization.ts +++ b/src/resolvers/Mutation/updateUserRoleInOrganization.ts @@ -4,11 +4,12 @@ import { ADMIN_CHANGING_ROLE_OF_CREATOR, ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ADMIN, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, USER_NOT_MEMBER_FOR_ORGANIZATION, } from "../../constants"; import { errors, requestContext } from "../../libraries"; -import { Organization, User } from "../../models"; +import { AppUserProfile, Organization, User } from "../../models"; import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; /** @@ -68,7 +69,16 @@ export const updateUserRoleInOrganization: MutationResolvers["updateUserRoleInOr USER_NOT_FOUND_ERROR.PARAM ); } - + const loggedInUserAppProfile = await AppUserProfile.findOne({ + userId: loggedInUser._id, + }).lean(); + if (!loggedInUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Check whether loggedIn user is admin of the organization. const loggedInUserIsOrganizationAdmin = organization?.admins.some((admin) => Types.ObjectId(admin).equals(loggedInUser._id) @@ -76,7 +86,7 @@ export const updateUserRoleInOrganization: MutationResolvers["updateUserRoleInOr if ( !loggedInUserIsOrganizationAdmin && - loggedInUser.userType !== "SUPERADMIN" + loggedInUserAppProfile.isSuperAdmin === false ) { throw new errors.NotFoundError( requestContext.translate(USER_NOT_AUTHORIZED_ADMIN.MESSAGE), @@ -87,7 +97,7 @@ export const updateUserRoleInOrganization: MutationResolvers["updateUserRoleInOr // Admin of Org cannot change the role of SUPERADMIN in an organization. if ( args.role === "SUPERADMIN" && - loggedInUser.userType !== "SUPERADMIN" && + !loggedInUserAppProfile.isSuperAdmin && loggedInUserIsOrganizationAdmin ) { throw new errors.NotFoundError( @@ -125,8 +135,8 @@ export const updateUserRoleInOrganization: MutationResolvers["updateUserRoleInOr $push: { admins: args.userId }, } ); - await User.updateOne( - { _id: args.userId }, + await AppUserProfile.updateOne( + { userId: args.userId }, { $push: { adminFor: args.organizationId } } ); return { ...organization, ...updatedOrg }; @@ -137,8 +147,8 @@ export const updateUserRoleInOrganization: MutationResolvers["updateUserRoleInOr $pull: { admins: args.userId }, } ).lean(); - await User.updateOne( - { _id: args.userId }, + await AppUserProfile.updateOne( + { userId: args.userId }, { $pull: { adminFor: args.organizationId } } ); return { ...organization, ...updatedOrg }; diff --git a/src/resolvers/Mutation/updateUserTag.ts b/src/resolvers/Mutation/updateUserTag.ts index 89269b79c6..a40557e27b 100644 --- a/src/resolvers/Mutation/updateUserTag.ts +++ b/src/resolvers/Mutation/updateUserTag.ts @@ -1,13 +1,14 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { errors, requestContext } from "../../libraries"; -import { User, OrganizationTagUser } from "../../models"; +import { Types } from "mongoose"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, - TAG_NOT_FOUND, NO_CHANGE_IN_TAG_NAME, TAG_ALREADY_EXISTS, + TAG_NOT_FOUND, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, OrganizationTagUser, User } from "../../models"; +import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; export const updateUserTag: MutationResolvers["updateUserTag"] = async ( _parent, @@ -26,6 +27,16 @@ export const updateUserTag: MutationResolvers["updateUserTag"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } // Get the tag object const existingTag = await OrganizationTagUser.findOne({ @@ -41,15 +52,16 @@ export const updateUserTag: MutationResolvers["updateUserTag"] = async ( } // Boolean to determine whether user is an admin of organization of the tag folder. - const currentUserIsOrganizationAdmin = currentUser.adminFor.some( - (organization) => organization.equals(existingTag?.organizationId) + const currentUserIsOrganizationAdmin = currentUserAppProfile.adminFor.some( + (organization) => + organization && + Types.ObjectId(organization.toString()).equals( + existingTag?.organizationId + ) ); // Checks whether currentUser can update the tag - if ( - !(currentUser.userType === "SUPERADMIN") && - !currentUserIsOrganizationAdmin - ) { + if (!currentUserAppProfile.isSuperAdmin && !currentUserIsOrganizationAdmin) { throw new errors.UnauthorizedError( requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), USER_NOT_AUTHORIZED_ERROR.CODE, @@ -58,7 +70,7 @@ export const updateUserTag: MutationResolvers["updateUserTag"] = async ( } // Throw error if the new tag name is the same as the old one - if (existingTag!.name === args.input.name) { + if (existingTag.name === args.input.name) { throw new errors.ConflictError( requestContext.translate(NO_CHANGE_IN_TAG_NAME.MESSAGE), NO_CHANGE_IN_TAG_NAME.CODE, @@ -69,8 +81,8 @@ export const updateUserTag: MutationResolvers["updateUserTag"] = async ( // Check if another tag with the new name exists under the same parent tag const anotherTagExists = await OrganizationTagUser.exists({ name: args.input.name, - parentTagId: existingTag!.parentTagId, - organizationId: existingTag!.organizationId, + parentTagId: existingTag.parentTagId, + organizationId: existingTag.organizationId, }); if (anotherTagExists) { diff --git a/src/resolvers/Mutation/updateUserType.ts b/src/resolvers/Mutation/updateUserType.ts deleted file mode 100644 index 63359df912..0000000000 --- a/src/resolvers/Mutation/updateUserType.ts +++ /dev/null @@ -1,69 +0,0 @@ -import type { MutationResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; -import { - USER_NOT_FOUND_ERROR, - SUPERADMIN_CANT_CHANGE_OWN_ROLE, -} from "../../constants"; -import { errors, requestContext } from "../../libraries"; -import { superAdminCheck } from "../../utilities"; -/** - * This function enables to update user type. - * @param _parent - parent of current request - * @param args - payload provided with the request - * @param context - context of entire application - * @remarks The following checks are done: - * 1. If the user exists. - * @returns Updated user type. - */ -export const updateUserType: MutationResolvers["updateUserType"] = async ( - _parent, - args, - context -) => { - const currentUser = await User.findOne({ - _id: context.userId, - }) - .select(["userType"]) - .lean(); - - if (!currentUser) { - throw new errors.NotFoundError( - requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), - USER_NOT_FOUND_ERROR.CODE, - USER_NOT_FOUND_ERROR.PARAM - ); - } - - superAdminCheck(currentUser); - if (args.data.id === currentUser._id.toString()) { - throw new errors.InputValidationError( - requestContext.translate(SUPERADMIN_CANT_CHANGE_OWN_ROLE.MESSAGE), - SUPERADMIN_CANT_CHANGE_OWN_ROLE.CODE, - SUPERADMIN_CANT_CHANGE_OWN_ROLE.PARAM - ); - } - - const userExists = await User.exists({ - _id: args.data.id!, - }); - - if (userExists === false) { - throw new errors.NotFoundError( - requestContext.translate(USER_NOT_FOUND_ERROR.MESSAGE), - USER_NOT_FOUND_ERROR.CODE, - USER_NOT_FOUND_ERROR.PARAM - ); - } - - await User.updateOne( - { - _id: args.data.id!, - }, - { - userType: args.data.userType!, - adminApproved: true, - } - ); - - return true; -}; diff --git a/src/resolvers/Query/checkAuth.ts b/src/resolvers/Query/checkAuth.ts index 93daebe0cc..4ea289acf7 100644 --- a/src/resolvers/Query/checkAuth.ts +++ b/src/resolvers/Query/checkAuth.ts @@ -1,6 +1,6 @@ import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; import { USER_NOT_FOUND_ERROR } from "../../constants"; -import { User } from "../../models"; +import { AppUserProfile, User } from "../../models"; import { errors } from "../../libraries"; /** * This query determines whether or not the user exists in the database (MongoDB). @@ -26,6 +26,16 @@ export const checkAuth: QueryResolvers["checkAuth"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + USER_NOT_FOUND_ERROR.DESC, + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); + } return { ...currentUser, diff --git a/src/resolvers/Query/helperFunctions/getSort.ts b/src/resolvers/Query/helperFunctions/getSort.ts index 75cc17c6b2..0a9382af79 100644 --- a/src/resolvers/Query/helperFunctions/getSort.ts +++ b/src/resolvers/Query/helperFunctions/getSort.ts @@ -15,7 +15,7 @@ export const getSort = ( | UserOrderByInput > | undefined -): any => { +): unknown => { let sortPayload = {}; switch (orderBy) { @@ -199,17 +199,17 @@ export const getSort = ( }; break; - case "appLanguageCode_ASC": - sortPayload = { - appLanguageCode: 1, - }; - break; + // case "appLanguageCode_ASC": + // sortPayload = { + // appLanguageCode: 1, + // }; + // break; - case "appLanguageCode_DESC": - sortPayload = { - appLanguageCode: -1, - }; - break; + // case "appLanguageCode_DESC": + // sortPayload = { + // appLanguageCode: -1, + // }; + // break; case "email_ASC": sortPayload = { diff --git a/src/resolvers/Query/helperFunctions/getWhere.ts b/src/resolvers/Query/helperFunctions/getWhere.ts index a9dfbe4613..565b2051a0 100644 --- a/src/resolvers/Query/helperFunctions/getWhere.ts +++ b/src/resolvers/Query/helperFunctions/getWhere.ts @@ -567,72 +567,72 @@ export const getWhere = ( } //Returns provided appLanguageCode user - if (where.appLanguageCode) { - wherePayload = { - ...wherePayload, - appLanguageCode: where.appLanguageCode, - }; - } - - //Returns user with not that provided appLanguageCode - if (where.appLanguageCode_not) { - wherePayload = { - ...wherePayload, - appLanguageCode: { - $ne: where.appLanguageCode_not, - }, - }; - } + // if (where.appLanguageCode) { + // wherePayload = { + // ...wherePayload, + // appLanguageCode: where.appLanguageCode, + // }; + // } + + // //Returns user with not that provided appLanguageCode + // if (where.appLanguageCode_not) { + // wherePayload = { + // ...wherePayload, + // appLanguageCode: { + // $ne: where.appLanguageCode_not, + // }, + // }; + // } // Objects appLanguageCode falls in provided list - if (where.appLanguageCode_in) { - wherePayload = { - ...wherePayload, - appLanguageCode: { - $in: where.appLanguageCode_in, - }, - }; - } - - // Return objects appLanguageCode not falls in the list - if (where.appLanguageCode_not_in) { - wherePayload = { - ...wherePayload, - appLanguageCode: { - $nin: where.appLanguageCode_not_in, - }, - }; - } - - // Return objects with appLanguageCode containing provided string - if (where.appLanguageCode_contains) { - wherePayload = { - ...wherePayload, - appLanguageCode: { - $regex: where.appLanguageCode_contains, - $options: "i", - }, - }; - } - - // Returns objects with appLanguageCode starts with provided string - if (where.appLanguageCode_starts_with) { - const regexp = new RegExp("^" + where.appLanguageCode_starts_with); - wherePayload = { - ...wherePayload, - appLanguageCode: regexp, - }; - } - - // Return users with admin for provided organizationId - if (where.admin_for) { - wherePayload = { - ...wherePayload, - adminFor: { - _id: where.admin_for, - }, - }; - } + // if (where.appLanguageCode_in) { + // wherePayload = { + // ...wherePayload, + // appLanguageCode: { + // $in: where.appLanguageCode_in, + // }, + // }; + // } + + // // Return objects appLanguageCode not falls in the list + // if (where.appLanguageCode_not_in) { + // wherePayload = { + // ...wherePayload, + // appLanguageCode: { + // $nin: where.appLanguageCode_not_in, + // }, + // }; + // } + + // // Return objects with appLanguageCode containing provided string + // if (where.appLanguageCode_contains) { + // wherePayload = { + // ...wherePayload, + // appLanguageCode: { + // $regex: where.appLanguageCode_contains, + // $options: "i", + // }, + // }; + // } + + // // Returns objects with appLanguageCode starts with provided string + // if (where.appLanguageCode_starts_with) { + // const regexp = new RegExp("^" + where.appLanguageCode_starts_with); + // wherePayload = { + // ...wherePayload, + // appLanguageCode: regexp, + // }; + // } + + // // Return users with admin for provided organizationId + // if (where.admin_for) { + // wherePayload = { + // ...wherePayload, + // adminFor: { + // _id: where.admin_for, + // }, + // }; + // } if (where.event_title_contains) { wherePayload = { diff --git a/src/resolvers/Query/myLanguage.ts b/src/resolvers/Query/myLanguage.ts index 2ebbc9f768..597ed871c6 100644 --- a/src/resolvers/Query/myLanguage.ts +++ b/src/resolvers/Query/myLanguage.ts @@ -1,7 +1,7 @@ -import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; -import { errors } from "../../libraries"; import { USER_NOT_FOUND_ERROR } from "../../constants"; +import { errors } from "../../libraries"; +import { AppUserProfile, User } from "../../models"; +import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; /** * This query fetch the current user language from the database. * @param _parent- @@ -17,9 +17,7 @@ export const myLanguage: QueryResolvers["myLanguage"] = async ( ) => { const currentUser = await User.findOne({ _id: context.userId, - }) - .select(["appLanguageCode"]) - .lean(); + }).lean(); if (!currentUser) { throw new errors.NotFoundError( @@ -28,6 +26,18 @@ export const myLanguage: QueryResolvers["myLanguage"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }) + .select(["appLanguageCode"]) + .lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthorizedError( + USER_NOT_FOUND_ERROR.MESSAGE, + USER_NOT_FOUND_ERROR.CODE, + USER_NOT_FOUND_ERROR.PARAM + ); + } - return currentUser.appLanguageCode; + return currentUserAppProfile.appLanguageCode; }; diff --git a/src/resolvers/Query/organizationsMemberConnection.ts b/src/resolvers/Query/organizationsMemberConnection.ts index 1f80254546..997ebad4d8 100644 --- a/src/resolvers/Query/organizationsMemberConnection.ts +++ b/src/resolvers/Query/organizationsMemberConnection.ts @@ -1,6 +1,6 @@ -import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; import type { InterfaceUser } from "../../models"; import { User } from "../../models"; +import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; import { getSort } from "./helperFunctions/getSort"; import { getWhere } from "./helperFunctions/getWhere"; @@ -15,32 +15,38 @@ import { getWhere } from "./helperFunctions/getWhere"; * @remarks Connection in graphQL means pagination, * learn more about Connection {@link https://relay.dev/graphql/connections.htm | here}. */ -// @ts-ignore export const organizationsMemberConnection: QueryResolvers["organizationsMemberConnection"] = async (_parent, args, context) => { const where = getWhere(args.where); const sort = getSort(args.orderBy); // Pagination based Options - let paginateOptions; + interface InterfacePaginateOptions { + lean?: boolean | undefined; + sort?: object | string | undefined; + pagination?: boolean | undefined; + page?: number | undefined; + limit?: number | undefined; + } + let paginateOptions: InterfacePaginateOptions = + {} as InterfacePaginateOptions; if (args.first) { if (args.skip === null) { throw "Missing Skip parameter. Set it to either 0 or some other value"; } - paginateOptions = { lean: true, sort: sort, pagination: true, page: args.skip, limit: args.first, - }; + } as InterfacePaginateOptions; } else { paginateOptions = { sort: sort, pagination: false, - }; + } as InterfacePaginateOptions; } const usersModel = await User.paginate( @@ -57,7 +63,7 @@ export const organizationsMemberConnection: QueryResolvers["organizationsMemberC } ); - let users = {}; + let users: InterfaceUser[] = []; // Change the type of users if (paginateOptions.pagination) { if (args.skip === undefined) { diff --git a/src/resolvers/Query/user.ts b/src/resolvers/Query/user.ts index efd0d191be..b9e634b6e0 100644 --- a/src/resolvers/Query/user.ts +++ b/src/resolvers/Query/user.ts @@ -1,7 +1,13 @@ import { USER_NOT_FOUND_ERROR } from "../../constants"; -import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; import { errors } from "../../libraries"; -import { User } from "../../models"; +import type { + InterfaceAppUserProfile, + InterfaceEvent, + InterfaceOrganization, + InterfaceUser, +} from "../../models"; +import { AppUserProfile, User } from "../../models"; +import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; /** * This query fetch the user from the database. * @param _parent- @@ -22,16 +28,35 @@ export const user: QueryResolvers["user"] = async (_parent, args, context) => { ); } - const user = await User.findOne({ + const user: InterfaceUser = await User.findOne({ _id: args.id, + }).lean(); + const userAppProfile: InterfaceAppUserProfile = await AppUserProfile.findOne({ + userId: user._id, }) .populate("adminFor") .lean(); // This Query field doesn't allow client to see organizations they are blocked by return { - ...user!, - image: user?.image ? `${context.apiRootUrl}${user.image}` : null, - organizationsBlockedBy: [], + user: { + ...user, + image: user?.image ? `${context.apiRootUrl}${user.image}` : null, + organizationsBlockedBy: [], + }, + appUserProfile: { + // ...userAppProfile, + _id: userAppProfile._id.toString(), + userId: userAppProfile.userId as InterfaceUser, + adminFor: userAppProfile.adminFor as InterfaceOrganization[], + appLanguageCode: userAppProfile.appLanguageCode, + isSuperAdmin: userAppProfile.isSuperAdmin, + pluginCreationAllowed: userAppProfile.pluginCreationAllowed, + tokenVersion: userAppProfile.tokenVersion, + eventAdmin: userAppProfile.eventAdmin as InterfaceEvent[], + createdEvents: userAppProfile.createdEvents as InterfaceEvent[], + createdOrganizations: + userAppProfile.createdOrganizations as InterfaceOrganization[], + }, }; }; diff --git a/src/resolvers/Query/userLanguage.ts b/src/resolvers/Query/userLanguage.ts index 3bcab4be00..b7ecd882d2 100644 --- a/src/resolvers/Query/userLanguage.ts +++ b/src/resolvers/Query/userLanguage.ts @@ -1,7 +1,10 @@ +import { + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import { AppUserProfile, User } from "../../models"; import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; -import { User } from "../../models"; -import { errors } from "../../libraries"; -import { USER_NOT_FOUND_ERROR } from "../../constants"; /** * This query will fetch the language code for the user from the database. * @param _parent- @@ -14,9 +17,7 @@ export const userLanguage: QueryResolvers["userLanguage"] = async ( ) => { const user = await User.findOne({ _id: args.userId, - }) - .select(["appLanguageCode"]) - .lean(); + }).lean(); if (!user) { throw new errors.NotFoundError( @@ -25,6 +26,18 @@ export const userLanguage: QueryResolvers["userLanguage"] = async ( USER_NOT_FOUND_ERROR.PARAM ); } + const appUserProfile = await AppUserProfile.findOne({ + userId: user._id, + }) + .select(["appLanguageCode"]) + .lean(); + if (!appUserProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ERROR.MESSAGE), + USER_NOT_AUTHORIZED_ERROR.CODE, + USER_NOT_AUTHORIZED_ERROR.PARAM + ); + } - return user.appLanguageCode; + return appUserProfile.appLanguageCode; }; diff --git a/src/resolvers/Query/users.ts b/src/resolvers/Query/users.ts index ab9fdfba43..43157e832c 100644 --- a/src/resolvers/Query/users.ts +++ b/src/resolvers/Query/users.ts @@ -1,8 +1,8 @@ -import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; -import type { InterfaceUser } from "../../models"; -import { User } from "../../models"; -import { errors, requestContext } from "../../libraries"; import { UNAUTHENTICATED_ERROR } from "../../constants"; +import { errors, requestContext } from "../../libraries"; +import type { InterfaceUser } from "../../models"; +import { AppUserProfile, User } from "../../models"; +import type { QueryResolvers } from "../../types/generatedGraphQLTypes"; import { getSort } from "./helperFunctions/getSort"; import { getWhere } from "./helperFunctions/getWhere"; @@ -33,9 +33,19 @@ export const users: QueryResolvers["users"] = async ( UNAUTHENTICATED_ERROR.PARAM ); } + const currentUserAppProfile = await AppUserProfile.findOne({ + userId: currentUser._id, + }).lean(); + if (!currentUserAppProfile) { + throw new errors.UnauthenticatedError( + requestContext.translate(UNAUTHENTICATED_ERROR.MESSAGE), + UNAUTHENTICATED_ERROR.CODE, + UNAUTHENTICATED_ERROR.PARAM + ); + } + const filterCriteria = { ...where, - ...(args.userType ? { userType: args.userType } : {}), }; if (args.adminApproved === true) { @@ -49,24 +59,20 @@ export const users: QueryResolvers["users"] = async ( .limit(args.first ?? 0) .skip(args.skip ?? 0) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .populate("organizationsBlockedBy") .lean(); return users.map((user) => { - const { userType } = currentUser; + const isSuperAdmin = currentUserAppProfile.isSuperAdmin; return { ...user, image: user.image ? `${context.apiRootUrl}${user.image}` : null, organizationsBlockedBy: - (userType === "ADMIN" || userType === "SUPERADMIN") && - currentUser._id !== user._id + isSuperAdmin && currentUser._id !== user._id ? user.organizationsBlockedBy : [], }; diff --git a/src/resolvers/Query/usersConnection.ts b/src/resolvers/Query/usersConnection.ts index 747712037a..1f5680329a 100644 --- a/src/resolvers/Query/usersConnection.ts +++ b/src/resolvers/Query/usersConnection.ts @@ -24,12 +24,8 @@ export const usersConnection: QueryResolvers["usersConnection"] = async ( .limit(args.first ?? 0) .skip(args.skip ?? 0) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); return users; diff --git a/src/typeDefs/enums.ts b/src/typeDefs/enums.ts index 9e9804c137..7d3c7c387c 100644 --- a/src/typeDefs/enums.ts +++ b/src/typeDefs/enums.ts @@ -90,8 +90,6 @@ export const enums = gql` lastName_DESC email_ASC email_DESC - appLanguageCode_ASC - appLanguageCode_DESC } enum UserType { diff --git a/src/typeDefs/inputs.ts b/src/typeDefs/inputs.ts index d0bdd94af4..517488db87 100644 --- a/src/typeDefs/inputs.ts +++ b/src/typeDefs/inputs.ts @@ -400,15 +400,6 @@ export const inputs = gql` email_contains: EmailAddress email_starts_with: EmailAddress - appLanguageCode: String - appLanguageCode_not: String - appLanguageCode_in: [String!] - appLanguageCode_not_in: [String!] - appLanguageCode_contains: String - appLanguageCode_starts_with: String - - admin_for: ID - event_title_contains: String } input PostUpdateInput { diff --git a/src/typeDefs/mutations.ts b/src/typeDefs/mutations.ts index a2439dccef..4f024488cd 100644 --- a/src/typeDefs/mutations.ts +++ b/src/typeDefs/mutations.ts @@ -48,9 +48,10 @@ export const mutations = gql` assignUserTag(input: ToggleUserTagAssignInput!): User @auth - blockPluginCreationBySuperadmin(userId: ID!, blockUser: Boolean!): User! - @auth - @role(requires: SUPERADMIN) + blockPluginCreationBySuperadmin( + userId: ID! + blockUser: Boolean! + ): AppUserProfile! @auth @role(requires: SUPERADMIN) blockUser(organizationId: ID!, userId: ID!): User! @auth @@ -60,7 +61,7 @@ export const mutations = gql` createMember(input: UserAndOrganizationInput!): Organization! @auth - createAdmin(data: UserAndOrganizationInput!): User! + createAdmin(data: UserAndOrganizationInput!): AppUserProfile! @auth @role(requires: SUPERADMIN) @@ -149,7 +150,7 @@ export const mutations = gql` rejectMembershipRequest(membershipRequestId: ID!): MembershipRequest! @auth - removeAdmin(data: UserAndOrganizationInput!): User! + removeAdmin(data: UserAndOrganizationInput!): AppUserProfile! @auth @role(requires: SUPERADMIN) @@ -172,7 +173,7 @@ export const mutations = gql` removeMember(data: UserAndOrganizationInput!): Organization! @auth - removeOrganization(id: ID!): User! @auth @role(requires: SUPERADMIN) + removeOrganization(id: ID!): UserData! @auth @role(requires: SUPERADMIN) removeOrganizationImage(organizationId: String!): Organization! @auth @@ -249,7 +250,7 @@ export const mutations = gql` updateUserProfile(data: UpdateUserInput, file: String): User! @auth - updateUserPassword(data: UpdateUserPasswordInput!): User! @auth + updateUserPassword(data: UpdateUserPasswordInput!): UserData! @auth updateUserRoleInOrganization( organizationId: ID! diff --git a/src/typeDefs/queries.ts b/src/typeDefs/queries.ts index 975f839ad9..6dc7511bad 100644 --- a/src/typeDefs/queries.ts +++ b/src/typeDefs/queries.ts @@ -100,7 +100,7 @@ export const queries = gql` registrantsByEvent(id: ID!): [User] - user(id: ID!): User! @auth + user(id: ID!): UserData! @auth userLanguage(userId: ID!): String @auth @@ -109,7 +109,7 @@ export const queries = gql` orderBy: UserOrderByInput first: Int skip: Int - userType: String + adminApproved: Boolean ): [User] @auth diff --git a/src/typeDefs/types.ts b/src/typeDefs/types.ts index 1923a9a37a..24324001d5 100644 --- a/src/typeDefs/types.ts +++ b/src/typeDefs/types.ts @@ -12,6 +12,7 @@ export const types = gql` type AuthData { user: User! + appUserProfile: AppUserProfile! accessToken: String! refreshToken: String! } @@ -402,29 +403,26 @@ export const types = gql` type User { _id: ID! + appUserProfileId: AppUserProfile address: Address adminApproved: Boolean - adminFor: [Organization] appLanguageCode: String! birthDate: Date createdAt: DateTime! - createdEvents: [Event] - createdOrganizations: [Organization] educationGrade: EducationGrade email: EmailAddress! employmentStatus: EmploymentStatus - eventAdmin: [Event] firstName: String! gender: Gender image: String joinedOrganizations: [Organization] lastName: String! maritalStatus: MaritalStatus - membershipRequests: [MembershipRequest] organizationsBlockedBy: [Organization] phone: UserPhone - pluginCreationAllowed: Boolean! + membershipRequests: [MembershipRequest] registeredEvents: [Event] + pluginCreationAllowed: Boolean! tagsAssignedWith( after: String before: String @@ -432,18 +430,28 @@ export const types = gql` last: PositiveInt organizationId: ID ): UserTagsConnection - tokenVersion: Int! updatedAt: DateTime! - userType: UserType! } - + type AppUserProfile { + _id: ID! + userId: User! + adminFor: [Organization] + createdEvents: [Event] + createdOrganizations: [Organization] + eventAdmin: [Event] + pluginCreationAllowed: Boolean! + isSuperAdmin: Boolean! + } type UserCustomData { _id: ID! organizationId: ID! userId: ID! values: JSON! } - + type UserData { + user: User! + appUserProfile: AppUserProfile! + } type UserConnection { pageInfo: PageInfo! edges: [User]! diff --git a/src/types/generatedGraphQLTypes.ts b/src/types/generatedGraphQLTypes.ts index 161abec318..b171f5ac6f 100644 --- a/src/types/generatedGraphQLTypes.ts +++ b/src/types/generatedGraphQLTypes.ts @@ -135,9 +135,22 @@ export type AggregateUser = { count: Scalars['Int']['output']; }; +export type AppUserProfile = { + __typename?: 'AppUserProfile'; + _id: Scalars['ID']['output']; + adminFor?: Maybe>>; + createdEvents?: Maybe>>; + createdOrganizations?: Maybe>>; + eventAdmin?: Maybe>>; + isSuperAdmin: Scalars['Boolean']['output']; + pluginCreationAllowed: Scalars['Boolean']['output']; + userId: User; +}; + export type AuthData = { __typename?: 'AuthData'; accessToken: Scalars['String']['output']; + appUserProfile: AppUserProfile; refreshToken: Scalars['String']['output']; user: User; }; @@ -592,13 +605,13 @@ export type Mutation = { adminRemoveEvent: Event; adminRemoveGroup: GroupChat; assignUserTag?: Maybe; - blockPluginCreationBySuperadmin: User; + blockPluginCreationBySuperadmin: AppUserProfile; blockUser: User; cancelMembershipRequest: MembershipRequest; checkIn: CheckIn; createActionItem: ActionItem; createActionItemCategory: ActionItemCategory; - createAdmin: User; + createAdmin: AppUserProfile; createAdvertisement: Advertisement; createComment?: Maybe; createDirectChat: DirectChat; @@ -629,7 +642,7 @@ export type Mutation = { rejectAdmin: Scalars['Boolean']['output']; rejectMembershipRequest: MembershipRequest; removeActionItem: ActionItem; - removeAdmin: User; + removeAdmin: AppUserProfile; removeAdvertisement?: Maybe; removeComment?: Maybe; removeDirectChat: DirectChat; @@ -637,7 +650,7 @@ export type Mutation = { removeEventAttendee: User; removeGroupChat: GroupChat; removeMember: Organization; - removeOrganization: User; + removeOrganization: UserData; removeOrganizationCustomField: OrganizationCustomField; removeOrganizationImage: Organization; removePost?: Maybe; @@ -668,7 +681,7 @@ export type Mutation = { updateOrganization: Organization; updatePluginStatus: Plugin; updatePost: Post; - updateUserPassword: User; + updateUserPassword: UserData; updateUserProfile: User; updateUserRoleInOrganization: Organization; updateUserTag?: Maybe; @@ -1455,7 +1468,7 @@ export type Query = { postsByOrganizationConnection?: Maybe; registeredEventsByUser?: Maybe>>; registrantsByEvent?: Maybe>>; - user: User; + user: UserData; userLanguage?: Maybe; users?: Maybe>>; usersConnection: Array>; @@ -1644,7 +1657,6 @@ export type QueryUsersArgs = { first?: InputMaybe; orderBy?: InputMaybe; skip?: InputMaybe; - userType?: InputMaybe; where?: InputMaybe; }; @@ -1794,16 +1806,13 @@ export type User = { _id: Scalars['ID']['output']; address?: Maybe
; adminApproved?: Maybe; - adminFor?: Maybe>>; appLanguageCode: Scalars['String']['output']; + appUserProfileId?: Maybe; birthDate?: Maybe; createdAt: Scalars['DateTime']['output']; - createdEvents?: Maybe>>; - createdOrganizations?: Maybe>>; educationGrade?: Maybe; email: Scalars['EmailAddress']['output']; employmentStatus?: Maybe; - eventAdmin?: Maybe>>; firstName: Scalars['String']['output']; gender?: Maybe; image?: Maybe; @@ -1816,9 +1825,7 @@ export type User = { pluginCreationAllowed: Scalars['Boolean']['output']; registeredEvents?: Maybe>>; tagsAssignedWith?: Maybe; - tokenVersion: Scalars['Int']['output']; updatedAt: Scalars['DateTime']['output']; - userType: UserType; }; @@ -1850,6 +1857,12 @@ export type UserCustomData = { values: Scalars['JSON']['output']; }; +export type UserData = { + __typename?: 'UserData'; + appUserProfile: AppUserProfile; + user: User; +}; + export type UserEdge = { __typename?: 'UserEdge'; cursor: Scalars['String']['output']; @@ -1875,8 +1888,6 @@ export type UserInput = { }; export type UserOrderByInput = - | 'appLanguageCode_ASC' - | 'appLanguageCode_DESC' | 'email_ASC' | 'email_DESC' | 'firstName_ASC' @@ -1950,13 +1961,6 @@ export type UserType = | 'USER'; export type UserWhereInput = { - admin_for?: InputMaybe; - appLanguageCode?: InputMaybe; - appLanguageCode_contains?: InputMaybe; - appLanguageCode_in?: InputMaybe>; - appLanguageCode_not?: InputMaybe; - appLanguageCode_not_in?: InputMaybe>; - appLanguageCode_starts_with?: InputMaybe; email?: InputMaybe; email_contains?: InputMaybe; email_in?: InputMaybe>; @@ -2103,7 +2107,8 @@ export type ResolversTypes = { AggregatePost: ResolverTypeWrapper; AggregateUser: ResolverTypeWrapper; Any: ResolverTypeWrapper; - AuthData: ResolverTypeWrapper & { user: ResolversTypes['User'] }>; + AppUserProfile: ResolverTypeWrapper & { adminFor?: Maybe>>, createdEvents?: Maybe>>, createdOrganizations?: Maybe>>, eventAdmin?: Maybe>>, userId: ResolversTypes['User'] }>; + AuthData: ResolverTypeWrapper & { appUserProfile: ResolversTypes['AppUserProfile'], user: ResolversTypes['User'] }>; Boolean: ResolverTypeWrapper; CheckIn: ResolverTypeWrapper; CheckInInput: CheckInInput; @@ -2212,6 +2217,7 @@ export type ResolversTypes = { UserAndOrganizationInput: UserAndOrganizationInput; UserConnection: ResolverTypeWrapper & { edges: Array> }>; UserCustomData: ResolverTypeWrapper; + UserData: ResolverTypeWrapper & { appUserProfile: ResolversTypes['AppUserProfile'], user: ResolversTypes['User'] }>; UserEdge: ResolverTypeWrapper & { node: ResolversTypes['User'] }>; UserFamily: ResolverTypeWrapper; UserInput: UserInput; @@ -2243,7 +2249,8 @@ export type ResolversParentTypes = { AggregatePost: AggregatePost; AggregateUser: AggregateUser; Any: Scalars['Any']['output']; - AuthData: Omit & { user: ResolversParentTypes['User'] }; + AppUserProfile: Omit & { adminFor?: Maybe>>, createdEvents?: Maybe>>, createdOrganizations?: Maybe>>, eventAdmin?: Maybe>>, userId: ResolversParentTypes['User'] }; + AuthData: Omit & { appUserProfile: ResolversParentTypes['AppUserProfile'], user: ResolversParentTypes['User'] }; Boolean: Scalars['Boolean']['output']; CheckIn: InterfaceCheckInModel; CheckInInput: CheckInInput; @@ -2341,6 +2348,7 @@ export type ResolversParentTypes = { UserAndOrganizationInput: UserAndOrganizationInput; UserConnection: Omit & { edges: Array> }; UserCustomData: UserCustomData; + UserData: Omit & { appUserProfile: ResolversParentTypes['AppUserProfile'], user: ResolversParentTypes['User'] }; UserEdge: Omit & { node: ResolversParentTypes['User'] }; UserFamily: InterfaceUserFamilyModel; UserInput: UserInput; @@ -2439,8 +2447,21 @@ export interface AnyScalarConfig extends GraphQLScalarTypeConfig = { + _id?: Resolver; + adminFor?: Resolver>>, ParentType, ContextType>; + createdEvents?: Resolver>>, ParentType, ContextType>; + createdOrganizations?: Resolver>>, ParentType, ContextType>; + eventAdmin?: Resolver>>, ParentType, ContextType>; + isSuperAdmin?: Resolver; + pluginCreationAllowed?: Resolver; + userId?: Resolver; + __isTypeOf?: IsTypeOfResolverFn; +}; + export type AuthDataResolvers = { accessToken?: Resolver; + appUserProfile?: Resolver; refreshToken?: Resolver; user?: Resolver; __isTypeOf?: IsTypeOfResolverFn; @@ -2740,13 +2761,13 @@ export type MutationResolvers>; adminRemoveGroup?: Resolver>; assignUserTag?: Resolver, ParentType, ContextType, RequireFields>; - blockPluginCreationBySuperadmin?: Resolver>; + blockPluginCreationBySuperadmin?: Resolver>; blockUser?: Resolver>; cancelMembershipRequest?: Resolver>; checkIn?: Resolver>; createActionItem?: Resolver>; createActionItemCategory?: Resolver>; - createAdmin?: Resolver>; + createAdmin?: Resolver>; createAdvertisement?: Resolver>; createComment?: Resolver, ParentType, ContextType, RequireFields>; createDirectChat?: Resolver>; @@ -2777,7 +2798,7 @@ export type MutationResolvers>; rejectMembershipRequest?: Resolver>; removeActionItem?: Resolver>; - removeAdmin?: Resolver>; + removeAdmin?: Resolver>; removeAdvertisement?: Resolver, ParentType, ContextType, RequireFields>; removeComment?: Resolver, ParentType, ContextType, RequireFields>; removeDirectChat?: Resolver>; @@ -2785,7 +2806,7 @@ export type MutationResolvers>; removeGroupChat?: Resolver>; removeMember?: Resolver>; - removeOrganization?: Resolver>; + removeOrganization?: Resolver>; removeOrganizationCustomField?: Resolver>; removeOrganizationImage?: Resolver>; removePost?: Resolver, ParentType, ContextType, RequireFields>; @@ -2816,7 +2837,7 @@ export type MutationResolvers>; updatePluginStatus?: Resolver>; updatePost?: Resolver>; - updateUserPassword?: Resolver>; + updateUserPassword?: Resolver>; updateUserProfile?: Resolver>; updateUserRoleInOrganization?: Resolver>; updateUserTag?: Resolver, ParentType, ContextType, RequireFields>; @@ -2966,7 +2987,7 @@ export type QueryResolvers, ParentType, ContextType, RequireFields>; registeredEventsByUser?: Resolver>>, ParentType, ContextType, Partial>; registrantsByEvent?: Resolver>>, ParentType, ContextType, RequireFields>; - user?: Resolver>; + user?: Resolver>; userLanguage?: Resolver, ParentType, ContextType, RequireFields>; users?: Resolver>>, ParentType, ContextType, Partial>; usersConnection?: Resolver>, ParentType, ContextType, Partial>; @@ -3018,16 +3039,13 @@ export type UserResolvers; address?: Resolver, ParentType, ContextType>; adminApproved?: Resolver, ParentType, ContextType>; - adminFor?: Resolver>>, ParentType, ContextType>; appLanguageCode?: Resolver; + appUserProfileId?: Resolver, ParentType, ContextType>; birthDate?: Resolver, ParentType, ContextType>; createdAt?: Resolver; - createdEvents?: Resolver>>, ParentType, ContextType>; - createdOrganizations?: Resolver>>, ParentType, ContextType>; educationGrade?: Resolver, ParentType, ContextType>; email?: Resolver; employmentStatus?: Resolver, ParentType, ContextType>; - eventAdmin?: Resolver>>, ParentType, ContextType>; firstName?: Resolver; gender?: Resolver, ParentType, ContextType>; image?: Resolver, ParentType, ContextType>; @@ -3040,9 +3058,7 @@ export type UserResolvers; registeredEvents?: Resolver>>, ParentType, ContextType>; tagsAssignedWith?: Resolver, ParentType, ContextType, Partial>; - tokenVersion?: Resolver; updatedAt?: Resolver; - userType?: Resolver; __isTypeOf?: IsTypeOfResolverFn; }; @@ -3061,6 +3077,12 @@ export type UserCustomDataResolvers; }; +export type UserDataResolvers = { + appUserProfile?: Resolver; + user?: Resolver; + __isTypeOf?: IsTypeOfResolverFn; +}; + export type UserEdgeResolvers = { cursor?: Resolver; node?: Resolver; @@ -3131,6 +3153,7 @@ export type Resolvers = { AggregatePost?: AggregatePostResolvers; AggregateUser?: AggregateUserResolvers; Any?: GraphQLScalarType; + AppUserProfile?: AppUserProfileResolvers; AuthData?: AuthDataResolvers; CheckIn?: CheckInResolvers; CheckInStatus?: CheckInStatusResolvers; @@ -3190,6 +3213,7 @@ export type Resolvers = { User?: UserResolvers; UserConnection?: UserConnectionResolvers; UserCustomData?: UserCustomDataResolvers; + UserData?: UserDataResolvers; UserEdge?: UserEdgeResolvers; UserFamily?: UserFamilyResolvers; UserPhone?: UserPhoneResolvers; diff --git a/src/utilities/adminCheck.ts b/src/utilities/adminCheck.ts index bd44cd28e1..558b69c366 100644 --- a/src/utilities/adminCheck.ts +++ b/src/utilities/adminCheck.ts @@ -1,8 +1,8 @@ import { Types } from "mongoose"; -import { errors, requestContext } from "../libraries"; import { USER_NOT_AUTHORIZED_ADMIN } from "../constants"; +import { errors, requestContext } from "../libraries"; import type { InterfaceOrganization } from "../models"; -import { User } from "../models"; +import { AppUserProfile } from "../models"; /** * If the current user is an admin of the organisation, this function returns `true` otherwise it returns `false`. * @remarks @@ -19,12 +19,17 @@ export const adminCheck = async ( (admin) => admin === userId || Types.ObjectId(admin).equals(userId) ); - const user = await User.findOne({ - _id: userId, - }); - const isUserSuperAdmin: boolean = user - ? user.userType === "SUPERADMIN" - : false; + const userAppProfile = await AppUserProfile.findOne({ + userId, + }).lean(); + if (!userAppProfile) { + throw new errors.UnauthorizedError( + requestContext.translate(USER_NOT_AUTHORIZED_ADMIN.MESSAGE), + USER_NOT_AUTHORIZED_ADMIN.CODE, + USER_NOT_AUTHORIZED_ADMIN.PARAM + ); + } + const isUserSuperAdmin: boolean = userAppProfile.isSuperAdmin; if (!userIsOrganizationAdmin && !isUserSuperAdmin) { throw new errors.UnauthorizedError( diff --git a/src/utilities/auth.ts b/src/utilities/auth.ts index d8db27a3a4..6562a9bd9e 100644 --- a/src/utilities/auth.ts +++ b/src/utilities/auth.ts @@ -1,6 +1,6 @@ import jwt from "jsonwebtoken"; import { ACCESS_TOKEN_SECRET, REFRESH_TOKEN_SECRET } from "../constants"; -import type { InterfaceUser } from "../models"; +import type { InterfaceAppUserProfile, InterfaceUser } from "../models"; import { User } from "../models"; export interface InterfaceJwtTokenPayload { @@ -16,10 +16,13 @@ export interface InterfaceJwtTokenPayload { * @param user - User data * @returns JSON Web Token string payload */ -export const createAccessToken = (user: InterfaceUser): string => { +export const createAccessToken = ( + user: InterfaceUser, + appUserProfile: InterfaceAppUserProfile +): string => { return jwt.sign( { - tokenVersion: user.tokenVersion, + tokenVersion: appUserProfile.tokenVersion, userId: user._id.toString(), firstName: user.firstName, lastName: user.lastName, @@ -32,14 +35,17 @@ export const createAccessToken = (user: InterfaceUser): string => { ); }; -export const createRefreshToken = (user: InterfaceUser): string => { +export const createRefreshToken = ( + user: InterfaceUser, + appUserProfile: InterfaceAppUserProfile +): string => { return jwt.sign( { - tokenVersion: user.tokenVersion, - userId: user._id.toString(), - firstName: user.firstName, - lastName: user.lastName, - email: user.email, + tokenVersion: appUserProfile?.tokenVersion, + userId: user?._id.toString(), + firstName: user?.firstName, + lastName: user?.lastName, + email: user?.email, }, REFRESH_TOKEN_SECRET as string, { diff --git a/src/utilities/createSampleOrganizationUtil.ts b/src/utilities/createSampleOrganizationUtil.ts index ee67fe7d73..b766970ba5 100644 --- a/src/utilities/createSampleOrganizationUtil.ts +++ b/src/utilities/createSampleOrganizationUtil.ts @@ -1,5 +1,12 @@ import type { InterfaceEvent, InterfacePost, InterfaceUser } from "../models"; -import { Organization, User, Event, Post, Plugin } from "../models"; +import { + AppUserProfile, + Event, + Organization, + Plugin, + Post, + User, +} from "../models"; import { faker } from "@faker-js/faker"; import type mongoose from "mongoose"; @@ -10,7 +17,7 @@ import { SampleData } from "../models/SampleData"; export const generateUserData = async ( organizationId: string, userType: string -): Promise> => { +) => { const gender: "male" | "female" = faker.helpers.arrayElement([ "male", "female", @@ -33,9 +40,17 @@ export const generateUserData = async ( )}.com`, password: "$2a$12$bSYpay6TRMpTOaAmYPFXku4avwmqfFBtmgg39TabxmtFEiz4plFtW", joinedOrganizations: [organizationId], - userType, + }); + + const appUserProfile = new AppUserProfile({ + userId: user._id, adminFor, }); + if (userType == "SUPERADMIN") { + appUserProfile.isSuperAdmin = true; + } + await appUserProfile.save(); + user.appUserProfileId = appUserProfile._id; await user.save(); @@ -43,21 +58,37 @@ export const generateUserData = async ( documentId: user._id, collectionName: "User", }); + const sampleModel2 = new SampleData({ + documentId: appUserProfile._id, + collectionName: "AppUserProfile", + }); await sampleModel.save(); - - return user; + await sampleModel2.save(); + return { + user, + appUserProfile, + }; }; const createUser = async ( generatedUser: InterfaceUser & mongoose.Document ): Promise> => { const savedUser = await generatedUser.save(); + const appUserProfile = await AppUserProfile.create({ + userId: savedUser._id, + }); const sampleModel = new SampleData({ documentId: savedUser._id, collectionName: "User", }); + const sampleModel2 = new SampleData({ + documentId: appUserProfile._id, + collectionName: "AppUserProfile", + }); + await sampleModel.save(); + await sampleModel2.save(); return savedUser; }; @@ -214,7 +245,9 @@ export const generateRandomPlugins = async ( export const createSampleOrganization = async (): Promise => { const _id = faker.database.mongodbObjectId(); - const creator = await generateUserData(_id, "ADMIN"); + const userData = await generateUserData(_id, "ADMIN"); + const creator = userData.user; + const creatorAppProfile = userData.appUserProfile; interface Address { city: string; @@ -257,15 +290,18 @@ export const createSampleOrganization = async (): Promise => { createdAt: Date.now(), }); - creator.adminFor.push(organization._id); + creatorAppProfile.adminFor.push(organization._id); - await creator.save(); + // await creator.save(); + await creatorAppProfile.save(); for (let j = 0; j < 10; j++) { const userType = j === 0 ? "ADMIN" : "USER"; const newUserData = await generateUserData(_id, userType); - const newUser = await createUser(newUserData); + + const newUser = newUserData.user; + const newUserAppProfile = newUserData.appUserProfile; organization.members.push(newUser._id); diff --git a/src/utilities/loadSampleData.ts b/src/utilities/loadSampleData.ts index eaa0a62158..28fe6cbab4 100644 --- a/src/utilities/loadSampleData.ts +++ b/src/utilities/loadSampleData.ts @@ -1,9 +1,9 @@ import "dotenv/config"; -import yargs from "yargs"; import fs from "fs/promises"; import path from "path"; +import yargs from "yargs"; import { connect } from "../db"; -import { User, Organization, Event, Post } from "../models"; +import { AppUserProfile, Event, Organization, Post, User } from "../models"; interface InterfaceArgs { items?: string; @@ -17,6 +17,7 @@ async function formatDatabase(): Promise { Organization.deleteMany({}), Event.deleteMany({}), Post.deleteMany({}), + AppUserProfile.deleteMany({}), ]); console.log("Cleared all collections\n"); } @@ -68,6 +69,9 @@ async function insertCollections(collections: string[]): Promise { case "posts": await Post.insertMany(docs); break; + case "appUserProfiles": + await AppUserProfile.insertMany(docs); + break; default: console.log("\x1b[31m", `Invalid collection name: ${collection}`); break; @@ -85,7 +89,13 @@ async function insertCollections(collections: string[]): Promise { } // Default collections available to insert -const collections = ["users", "organizations", "posts", "events"]; +const collections = [ + "users", + "organizations", + "posts", + "events", + "appUserProfiles", +]; // Check if specific collections need to be inserted const { items: argvItems } = yargs diff --git a/src/utilities/removeSampleOrganizationUtil.ts b/src/utilities/removeSampleOrganizationUtil.ts index bbd14c9488..460507836e 100644 --- a/src/utilities/removeSampleOrganizationUtil.ts +++ b/src/utilities/removeSampleOrganizationUtil.ts @@ -1,4 +1,12 @@ -import { User, Organization, Post, Event, Plugin, SampleData } from "../models"; +import { + AppUserProfile, + Event, + Organization, + Plugin, + Post, + SampleData, + User, +} from "../models"; export async function removeSampleOrganization(): Promise { const sampleDataDocuments = await SampleData.find({}); @@ -11,6 +19,7 @@ export async function removeSampleOrganization(): Promise { Event, User, Plugin, + AppUserProfile, }; const collectionModel = collectionModels[collectionName]; diff --git a/src/utilities/superAdminCheck.ts b/src/utilities/superAdminCheck.ts index 1b6c52f580..64c0f06bce 100644 --- a/src/utilities/superAdminCheck.ts +++ b/src/utilities/superAdminCheck.ts @@ -1,9 +1,11 @@ -import { errors, requestContext } from "../libraries"; import { USER_NOT_AUTHORIZED_SUPERADMIN } from "../constants"; -import type { InterfaceUser } from "../models"; +import { errors, requestContext } from "../libraries"; +import type { InterfaceAppUserProfile } from "../models"; -export const superAdminCheck = (user: InterfaceUser): void => { - const userIsSuperAdmin: boolean = user.userType === "SUPERADMIN"; +export const superAdminCheck = ( + appUserProfile: InterfaceAppUserProfile +): void => { + const userIsSuperAdmin: boolean = appUserProfile.isSuperAdmin; if (!userIsSuperAdmin) { throw new errors.UnauthorizedError( diff --git a/src/utilities/userFamilyAdminCheck.ts b/src/utilities/userFamilyAdminCheck.ts index bf482733bf..5a0b700d37 100644 --- a/src/utilities/userFamilyAdminCheck.ts +++ b/src/utilities/userFamilyAdminCheck.ts @@ -1,8 +1,8 @@ import { Types } from "mongoose"; -import { errors, requestContext } from "../libraries"; import { USER_NOT_AUTHORIZED_ADMIN } from "../constants"; +import { errors, requestContext } from "../libraries"; +import { AppUserProfile } from "../models"; import type { InterfaceUserFamily } from "../models/userFamily"; -import { User } from "../models"; /** * If the current user is an admin of the organisation, this function returns `true` otherwise it returns `false`. * @remarks @@ -19,12 +19,13 @@ export const adminCheck = async ( (admin) => admin === userId || Types.ObjectId(admin).equals(userId) ); - const user = await User.findOne({ - _id: userId, + // const user = await User.findOne({ + // _id: userId, + // }); + const appUserProfile = await AppUserProfile.findOne({ + userId: userId, }); - const isUserSuperAdmin: boolean = user - ? user.userType === "SUPERADMIN" - : false; + const isUserSuperAdmin: boolean = appUserProfile?.isSuperAdmin || false; if (!userIsUserFamilyAdmin && !isUserSuperAdmin) { throw new errors.UnauthorizedError( diff --git a/tests/directives/directiveTransformer/roleDirectiveTransformer.spec.ts b/tests/directives/directiveTransformer/roleDirectiveTransformer.spec.ts index 204eafd918..68ce66f935 100644 --- a/tests/directives/directiveTransformer/roleDirectiveTransformer.spec.ts +++ b/tests/directives/directiveTransformer/roleDirectiveTransformer.spec.ts @@ -1,22 +1,21 @@ -import type { InterfaceUser } from "../../../src/models"; -import { User } from "../../../src/models"; -import { beforeAll, afterAll, it, expect } from "vitest"; -import { connect, disconnect } from "../../helpers/db"; -import type { Document } from "mongoose"; -import type mongoose from "mongoose"; -import { Types } from "mongoose"; import { ApolloServer } from "@apollo/server"; -import { gql } from "graphql-tag"; -import { errors } from "../../../src/libraries"; -import { nanoid } from "nanoid"; +import { makeExecutableSchema } from "@graphql-tools/schema"; import "dotenv/config"; -import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; -import i18n from "i18n"; import express from "express"; +import { gql } from "graphql-tag"; +import i18n from "i18n"; +import type mongoose from "mongoose"; +import { Types } from "mongoose"; +import { nanoid } from "nanoid"; +import { afterAll, beforeAll, expect, it } from "vitest"; import { appConfig } from "../../../src/config"; -import { makeExecutableSchema } from "@graphql-tools/schema"; +import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; import authDirectiveTransformer from "../../../src/directives/directiveTransformer/authDirectiveTransformer"; import roleDirectiveTransformer from "../../../src/directives/directiveTransformer/roleDirectiveTransformer"; +import { errors } from "../../../src/libraries"; +import { User } from "../../../src/models"; +import { connect, disconnect } from "../../helpers/db"; +import type { TestUserType } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -39,7 +38,7 @@ i18n.configure({ }); app.use(i18n.init); -let testUser: InterfaceUser & Document; +let testUser: TestUserType; const typeDefs = gql` directive @role(requires: String) on FIELD_DEFINITION @@ -70,7 +69,7 @@ beforeAll(async () => { }); afterAll(async () => { - await testUser.remove(); + await testUser?.remove(); await disconnect(MONGOOSE_INSTANCE); }); @@ -82,7 +81,7 @@ it("throws NotFoundError if no user exists with _id === context.userId", async ( `; const authenticatedContext = { userId: Types.ObjectId().toString(), - userType: testUser.userType, + userType: testUser?.userType, }; let schema = makeExecutableSchema({ typeDefs, @@ -111,121 +110,3 @@ it("throws NotFoundError if no user exists with _id === context.userId", async ( } } }); - -it("throws UnauthenticatedError if user exists but userType != requires", async () => { - const query = ` - query { - hello - } - `; - testUser.userType = "USER"; - await testUser.save(); - const authenticatedContext = { - userId: testUser._id, - userType: testUser.userType, - }; - let schema = makeExecutableSchema({ - typeDefs, - resolvers, - }); - // defines directives - schema = authDirectiveTransformer(schema, "auth"); - schema = roleDirectiveTransformer(schema, "role"); - const apolloServer = new ApolloServer({ - schema, - }); - - try { - await apolloServer.executeOperation( - { - query, - variables: {}, - }, - { - contextValue: authenticatedContext, - } - ); - - //@ts-ignore - expect(result.body.singleResult.data).toEqual({ hello: "hi" }); - } catch (err) { - if (err instanceof errors.UnauthenticatedError) { - expect(err.message).toEqual("user.notAuthenticated"); - } - } -}); - -it("returns data if user exists and userType === requires", async () => { - const query = ` - query { - hello - } - `; - testUser.userType = "ADMIN"; - await testUser.save(); - const authenticatedContext = { - userId: testUser._id, - userType: testUser.userType, - }; - let schema = makeExecutableSchema({ - typeDefs, - resolvers, - }); - // defines directives - schema = authDirectiveTransformer(schema, "auth"); - schema = roleDirectiveTransformer(schema, "role"); - const apolloServer = new ApolloServer({ - schema, - }); - - const result = await apolloServer.executeOperation( - { - query, - variables: {}, - }, - { - contextValue: authenticatedContext, - } - ); - - //@ts-ignore - - expect(result.body.singleResult.data).toEqual({ hello: "hi" }); -}); - -it("checks if the resolver is supplied, and return null data, if not", async () => { - const query = ` - query { - hello - } - `; - testUser.userType = "ADMIN"; - await testUser.save(); - const authenticatedContext = { - userId: testUser._id, - userType: testUser.userType, - }; - let schema = makeExecutableSchema({ - typeDefs, - resolvers, - }); - // defines directives - schema = authDirectiveTransformer(schema, "auth"); - schema = roleDirectiveTransformer(schema, "role"); - const apolloServer = new ApolloServer({ - schema, - }); - - const result = await apolloServer.executeOperation( - { - query, - variables: {}, - }, - { - contextValue: authenticatedContext, - } - ); - //@ts-ignore - - expect(result.body.singleResult.data).toEqual({ hello: "hi" }); -}); diff --git a/tests/helpers/advertisement.ts b/tests/helpers/advertisement.ts index e169a33e4f..cec6f2fbac 100644 --- a/tests/helpers/advertisement.ts +++ b/tests/helpers/advertisement.ts @@ -1,8 +1,8 @@ -import { Advertisement, User } from "../../src/models"; import type { InterfaceUser } from "../../src/models"; +import { Advertisement, AppUserProfile, User } from "../../src/models"; -import { nanoid } from "nanoid"; import type { Document } from "mongoose"; +import { nanoid } from "nanoid"; export type TestAdvertisementType = { _id: string; @@ -37,7 +37,7 @@ export const createTestAdvertisement = }; export type TestSuperAdminType = - | (InterfaceUser & Document) + | (InterfaceUser & Document) | null; export const createTestSuperAdmin = async (): Promise => { @@ -47,9 +47,22 @@ export const createTestSuperAdmin = async (): Promise => { firstName: `firstName${nanoid().toLowerCase()}`, lastName: `lastName${nanoid().toLowerCase()}`, image: null, + }); + const testSuperAdminAppProfile = await AppUserProfile.create({ + userId: testSuperAdmin._id, appLanguageCode: "en", - userType: "SUPERADMIN", // Set userType to "SUPERADMIN" + isSuperAdmin: true, }); + await User.updateOne( + { + _id: testSuperAdmin._id, + }, + { + $set: { + appUserProfileId: testSuperAdminAppProfile._id, + }, + } + ); return testSuperAdmin; }; diff --git a/tests/helpers/checkIn.ts b/tests/helpers/checkIn.ts index ae82ee1a5f..4dfa97034f 100644 --- a/tests/helpers/checkIn.ts +++ b/tests/helpers/checkIn.ts @@ -5,10 +5,10 @@ import type { Document } from "mongoose"; import { createTestEventWithRegistrants } from "./eventsWithRegistrants"; import type { TestOrganizationType, TestUserType } from "./userAndOrg"; export type TestEventType = - | (InterfaceEvent & Document) + | (InterfaceEvent & Document) | null; export type TestCheckInType = - | (InterfaceCheckIn & Document) + | (InterfaceCheckIn & Document) | null; export const createEventWithCheckedInUser = async (): Promise< @@ -17,12 +17,12 @@ export const createEventWithCheckedInUser = async (): Promise< const [testUser, testOrg, testEvent] = await createTestEventWithRegistrants(); const eventAttendee = await EventAttendee.findOne({ - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id, }).lean(); const checkIn = await CheckIn.create({ - eventAttendeeId: eventAttendee!._id, + eventAttendeeId: eventAttendee?._id, allotedRoom: nanoid(), allotedSeat: nanoid(), time: new Date(), @@ -30,11 +30,11 @@ export const createEventWithCheckedInUser = async (): Promise< await EventAttendee.updateOne( { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id, }, { - checkInId: checkIn!._id, + checkInId: checkIn?._id, } ); diff --git a/tests/helpers/events.ts b/tests/helpers/events.ts index 53432afa53..49a4c6bed9 100644 --- a/tests/helpers/events.ts +++ b/tests/helpers/events.ts @@ -1,12 +1,12 @@ -import type { TestOrganizationType, TestUserType } from "./userAndOrg"; -import { createTestUserAndOrganization } from "./userAndOrg"; -import type { InterfaceEvent } from "../../src/models"; -import { Event, EventAttendee, User } from "../../src/models"; import type { Document } from "mongoose"; import { nanoid } from "nanoid"; +import type { InterfaceEvent } from "../../src/models"; +import { AppUserProfile, Event, EventAttendee, User } from "../../src/models"; +import type { TestOrganizationType, TestUserType } from "./userAndOrg"; +import { createTestUserAndOrganization } from "./userAndOrg"; export type TestEventType = - | (InterfaceEvent & Document) + | (InterfaceEvent & Document) | null; export const createTestEvent = async (): Promise< @@ -34,11 +34,20 @@ export const createTestEvent = async (): Promise< { _id: testUser._id, }, + { + $push: { + registeredEvents: testEvent._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: testUser._id, + }, { $push: { eventAdmin: testEvent._id, createdEvents: testEvent._id, - registeredEvents: testEvent._id, }, } ); @@ -74,18 +83,27 @@ export const createEventWithRegistrant = async ( await EventAttendee.create({ userId, - eventId: testEvent!._id, + eventId: testEvent._id, }); await User.updateOne( { _id: userId, }, + { + $push: { + registeredEvents: testEvent._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId, + }, { $push: { eventAdmin: testEvent._id, createdEvents: testEvent._id, - registeredEvents: testEvent._id, }, } ); diff --git a/tests/helpers/eventsWithRegistrants.ts b/tests/helpers/eventsWithRegistrants.ts index f3ff4dca31..01503a0ec5 100644 --- a/tests/helpers/eventsWithRegistrants.ts +++ b/tests/helpers/eventsWithRegistrants.ts @@ -1,11 +1,11 @@ +import type { Document } from "mongoose"; +import type { InterfaceEvent } from "../../src/models"; +import { AppUserProfile, Event, EventAttendee, User } from "../../src/models"; import type { TestOrganizationType, TestUserType } from "./userAndOrg"; import { createTestUserAndOrganization } from "./userAndOrg"; -import type { InterfaceEvent } from "../../src/models"; -import { Event, EventAttendee, User } from "../../src/models"; -import type { Document } from "mongoose"; export type TestEventType = - | (InterfaceEvent & Document) + | (InterfaceEvent & Document) | null; export const createTestEventWithRegistrants = async ( @@ -14,9 +14,9 @@ export const createTestEventWithRegistrants = async ( const [testUser, testOrganization] = await createTestUserAndOrganization(); const testEvent = await Event.create({ - creatorId: testUser!._id, - admins: [testUser!._id], - organization: testOrganization!._id, + creatorId: testUser?._id, + admins: [testUser?._id], + organization: testOrganization?._id, isRegisterable: true, isPublic: true, title: "title", @@ -26,19 +26,28 @@ export const createTestEventWithRegistrants = async ( }); await EventAttendee.create({ - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id, }); await User.updateOne( { - _id: testUser!._id, + _id: testUser?._id, + }, + { + $push: { + registeredEvents: testEvent._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: testUser?._id, }, { $push: { eventAdmin: isAdmin ? testEvent._id : [], createdEvents: testEvent._id, - registeredEvents: testEvent._id, }, } ); diff --git a/tests/helpers/feedback.ts b/tests/helpers/feedback.ts index e48123bbd2..2f9a18a5c3 100644 --- a/tests/helpers/feedback.ts +++ b/tests/helpers/feedback.ts @@ -1,12 +1,12 @@ -import { nanoid } from "nanoid"; -import { CheckIn, type InterfaceFeedback, Feedback } from "../../src/models"; import type { Document } from "mongoose"; +import { nanoid } from "nanoid"; +import { CheckIn, Feedback, type InterfaceFeedback } from "../../src/models"; +import type { TestCheckInType, TestEventType } from "./checkIn"; import { createEventWithCheckedInUser } from "./checkIn"; -import type { TestEventType, TestCheckInType } from "./checkIn"; import type { TestOrganizationType, TestUserType } from "./userAndOrg"; export type TestFeedbackType = - | (InterfaceFeedback & Document) + | (InterfaceFeedback & Document) | null; export const createFeedbackWithIDs = async ( @@ -36,7 +36,10 @@ export const createFeedback = async (): Promise< ] > => { const result = await createEventWithCheckedInUser(); - const feedback = await createFeedbackWithIDs(result[2]!._id, result[3]!._id); + const feedback = await createFeedbackWithIDs( + result[2]?._id.toString() ?? "", + result[3]?._id.toString() ?? "" + ); return [...result, feedback]; }; diff --git a/tests/helpers/membershipRequests.ts b/tests/helpers/membershipRequests.ts index 05a4f4fbf8..a9d9250132 100644 --- a/tests/helpers/membershipRequests.ts +++ b/tests/helpers/membershipRequests.ts @@ -1,13 +1,18 @@ import type { TestUserType, TestOrganizationType } from "./userAndOrg"; import { createTestUser, createTestUserAndOrganization } from "./userAndOrg"; import type { InterfaceMembershipRequest } from "../../src/models"; -import { MembershipRequest, Organization, User } from "../../src/models"; +import { + AppUserProfile, + MembershipRequest, + Organization, + User, +} from "../../src/models"; import type { Document } from "mongoose"; import { nanoid } from "nanoid"; export type TestMembershipRequestType = | (InterfaceMembershipRequest & - Document) + Document) | null; export const createTestMembershipRequest = async (): Promise< @@ -34,11 +39,20 @@ export const createTestMembershipRequest = async (): Promise< { _id: testUser._id, }, + { + $push: { + membershipRequests: testMembershipRequest._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: testUser._id, + }, { $push: { createdOrganizations: testOrganization._id, adminFor: testOrganization._id, - membershipRequests: testMembershipRequest._id, }, } ); diff --git a/tests/helpers/tags.ts b/tests/helpers/tags.ts index ab9a997b8b..536b4eccbc 100644 --- a/tests/helpers/tags.ts +++ b/tests/helpers/tags.ts @@ -1,8 +1,8 @@ +import { nanoid } from "nanoid"; import type { InterfaceOrganizationTagUser } from "../../src/models"; import { OrganizationTagUser, TagUser } from "../../src/models"; -import { nanoid } from "nanoid"; -import type { TestUserType, TestOrganizationType } from "./userAndOrg"; -import { createTestUserAndOrganization, createTestUser } from "./userAndOrg"; +import type { TestOrganizationType, TestUserType } from "./userAndOrg"; +import { createTestUser, createTestUserAndOrganization } from "./userAndOrg"; export type TestUserTagType = InterfaceOrganizationTagUser | null; @@ -87,8 +87,8 @@ export const createTagsAndAssignToUser = async ( ): Promise<[TestUserType, TestOrganizationType, TestUserTagType[]]> => { const [testUser, testOrg, testTag] = await createRootTagWithOrg(); await TagUser.create({ - userId: testUser!._id, - tagId: testTag!._id, + userId: testUser?._id ?? "", + tagId: testTag?._id ?? "", }); const tags: TestUserTagType[] = [testTag]; @@ -101,8 +101,8 @@ export const createTagsAndAssignToUser = async ( tags.push(newTag.toObject()); await TagUser.create({ - tagId: newTag!._id, - userId: testUser!._id, + tagId: newTag?._id ?? "", + userId: testUser?._id ?? "", }); } diff --git a/tests/helpers/user.ts b/tests/helpers/user.ts index d05bb3a3eb..a660873179 100644 --- a/tests/helpers/user.ts +++ b/tests/helpers/user.ts @@ -1,10 +1,10 @@ -import type { InterfaceUser } from "../../src/models"; -import { User } from "../../src/models"; -import { nanoid } from "nanoid"; import type { Document } from "mongoose"; +import { nanoid } from "nanoid"; +import type { InterfaceUser } from "../../src/models"; +import { AppUserProfile, User } from "../../src/models"; export type TestUserType = - | (InterfaceUser & Document) + | (InterfaceUser & Document) | null; export const createTestUser = async (): Promise => { @@ -13,9 +13,19 @@ export const createTestUser = async (): Promise => { password: `pass${nanoid().toLowerCase()}`, firstName: `firstName${nanoid().toLowerCase()}`, lastName: `lastName${nanoid().toLowerCase()}`, - appLanguageCode: "en", }); - + const testUserAppProfile = await AppUserProfile.create({ + userId: testUser._id, + languageCode: "en", + }); + await User.updateOne( + { + _id: testUser._id, + }, + { + appUserProfileId: testUserAppProfile._id, + } + ); return testUser; }; diff --git a/tests/helpers/userAndOrg.ts b/tests/helpers/userAndOrg.ts index d3e7c11f73..1d617b852d 100644 --- a/tests/helpers/userAndOrg.ts +++ b/tests/helpers/userAndOrg.ts @@ -1,7 +1,12 @@ -import type { InterfaceOrganization, InterfaceUser } from "../../src/models"; -import { Organization, User } from "../../src/models"; -import { nanoid } from "nanoid"; +/* eslint-disable @typescript-eslint/no-explicit-any */ import type { Document } from "mongoose"; +import { nanoid } from "nanoid"; +import type { + InterfaceAppUserProfile, + InterfaceOrganization, + InterfaceUser, +} from "../../src/models"; +import { AppUserProfile, Organization, User } from "../../src/models"; export type TestOrganizationType = | (InterfaceOrganization & Document) @@ -10,16 +15,32 @@ export type TestOrganizationType = export type TestUserType = | (InterfaceUser & Document) | null; - +export type TestAppUserProfileType = + | (InterfaceAppUserProfile & Document) + | null; export const createTestUser = async (): Promise => { - const testUser = await User.create({ + let testUser = await User.create({ email: `email${nanoid().toLowerCase()}@gmail.com`, password: `pass${nanoid().toLowerCase()}`, firstName: `firstName${nanoid().toLowerCase()}`, lastName: `lastName${nanoid().toLowerCase()}`, image: null, + }); + const testUserAppProfile = await AppUserProfile.create({ + userId: testUser._id, appLanguageCode: "en", }); + testUser = (await User.findOneAndUpdate( + { + _id: testUser._id, + }, + { + appUserProfileId: testUserAppProfile._id, + }, + { + new: true, + } + )) as InterfaceUser & Document; return testUser; }; @@ -44,15 +65,23 @@ export const createTestOrganizationWithAdmin = async ( { _id: userID, }, + { + $push: { + joinedOrganizations: testOrganization._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: userID, + }, { $push: { createdOrganizations: testOrganization._id, adminFor: testOrganization._id, - joinedOrganizations: testOrganization._id, }, } ); - return testOrganization; }; @@ -90,11 +119,20 @@ export const createOrganizationwithVisibility = async ( { _id: userID, }, + { + $push: { + joinedOrganizations: testOrganization._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: userID, + }, { $push: { createdOrganizations: testOrganization._id, adminFor: testOrganization._id, - joinedOrganizations: testOrganization._id, }, } ); diff --git a/tests/helpers/userAndUserFamily.ts b/tests/helpers/userAndUserFamily.ts index 561f38cf9d..c39ef31755 100644 --- a/tests/helpers/userAndUserFamily.ts +++ b/tests/helpers/userAndUserFamily.ts @@ -1,8 +1,8 @@ import { nanoid } from "nanoid"; +import type { InterfaceUser } from "../../src/models"; +import { AppUserProfile, User } from "../../src/models"; import type { InterfaceUserFamily } from "../../src/models/userFamily"; -import { User } from "../../src/models"; import { UserFamily } from "../../src/models/userFamily"; -import type { InterfaceUser } from "../../src/models"; import type { Document } from "mongoose"; /* eslint-disable */ @@ -21,7 +21,10 @@ export const createTestUserFunc = async (): Promise => { firstName: `firstName${nanoid().toLowerCase()}`, lastName: `lastName${nanoid().toLowerCase()}`, appLanguageCode: "en", - userType: "SUPERADMIN", + }); + await AppUserProfile.create({ + userId: testUser._id, + isSuperAdmin: true, }); return testUser; diff --git a/tests/resolvers/Event/averageFeedbackScore.test.ts b/tests/resolvers/Event/averageFeedbackScore.test.ts index ed6bdebad3..21a4ed2a8d 100644 --- a/tests/resolvers/Event/averageFeedbackScore.test.ts +++ b/tests/resolvers/Event/averageFeedbackScore.test.ts @@ -1,14 +1,14 @@ import "dotenv/config"; -import { averageFeedbackScore as averageFeedbackScoreResolver } from "../../../src/resolvers/Event/averageFeedbackScore"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; -import type { TestEventType } from "../../helpers/events"; -import { createFeedbackWithIDs } from "../../helpers/feedback"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; +import { averageFeedbackScore as averageFeedbackScoreResolver } from "../../../src/resolvers/Event/averageFeedbackScore"; import { - type TestCheckInType, createEventWithCheckedInUser, + type TestCheckInType, } from "../../helpers/checkIn"; +import { connect, disconnect } from "../../helpers/db"; +import type { TestEventType } from "../../helpers/events"; +import { createFeedbackWithIDs } from "../../helpers/feedback"; let MONGOOSE_INSTANCE: typeof mongoose; let testEvent: TestEventType; @@ -25,7 +25,8 @@ afterAll(async () => { describe("resolvers -> Event -> averageFeedbackScore", () => { it(`Should return 0 if there are no submitted feedbacks for the event`, async () => { - const parent = testEvent!.toObject(); + if (!testEvent) throw new Error("testEvent is null"); + const parent = testEvent.toObject(); const averageFeedbackScorePayload = await averageFeedbackScoreResolver?.( parent, @@ -37,9 +38,14 @@ describe("resolvers -> Event -> averageFeedbackScore", () => { }); it(`Should return the proper average score if there are some submitted feedbacks for the event`, async () => { - await createFeedbackWithIDs(testEvent!._id, testCheckIn!._id); + if (!testEvent) throw new Error("testEvent is null"); + if (!testCheckIn) throw new Error("testCheckIn is null"); + await createFeedbackWithIDs( + testEvent._id.toString(), + testCheckIn._id.toString() + ); - const parent = testEvent!.toObject(); + const parent = testEvent.toObject(); const averageFeedbackScorePayload = await averageFeedbackScoreResolver?.( parent, diff --git a/tests/resolvers/Mutation/acceptAdmin.spec.ts b/tests/resolvers/Mutation/acceptAdmin.spec.ts index c6540c6711..a59bb667ee 100644 --- a/tests/resolvers/Mutation/acceptAdmin.spec.ts +++ b/tests/resolvers/Mutation/acceptAdmin.spec.ts @@ -4,11 +4,6 @@ import { Types } from "mongoose"; import type { MutationAcceptAdminArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { acceptAdmin as acceptAdminResolver } from "../../../src/resolvers/Mutation/acceptAdmin"; -import { - USER_NOT_AUTHORIZED_SUPERADMIN, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; import { afterAll, afterEach, @@ -18,9 +13,14 @@ import { it, vi, } from "vitest"; +import { + USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile, User } from "../../../src/models"; +import { acceptAdmin as acceptAdminResolver } from "../../../src/resolvers/Mutation/acceptAdmin"; import type { TestUserType } from "../../helpers/userAndOrg"; import { createTestUser } from "../../helpers/userAndOrg"; -import { User } from "../../../src/models"; let testUserSuperAdmin: TestUserType; let testUserAdmin: TestUserType; @@ -62,9 +62,9 @@ describe("resolvers -> Mutation -> acceptAdmin", () => { "../../../src/resolvers/Mutation/acceptAdmin" ); await acceptAdmin?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -89,23 +89,32 @@ describe("resolvers -> Mutation -> acceptAdmin", () => { "../../../src/resolvers/Mutation/acceptAdmin" ); await acceptAdmin?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE}` ); } }); it(`makes user with _id === args.id adminApproved and returns true`, async () => { + await AppUserProfile.updateOne( + { + userId: testUserSuperAdmin?._id, + }, + { + $set: { + isSuperAdmin: true, + }, + } + ); await User.updateOne( { - _id: testUserSuperAdmin?._id, + _id: testUserAdmin?._id, }, { $set: { - userType: "SUPERADMIN", - adminApproved: true, + adminApproved: false, }, } ); @@ -150,11 +159,40 @@ describe("resolvers -> Mutation -> acceptAdmin", () => { "../../../src/resolvers/Mutation/acceptAdmin" ); await acceptAdmin?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } }); + it("throws an error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementation((message) => `Translated ${message}`); + + try { + const testUser = await createTestUser(); + const args: MutationAcceptAdminArgs = { + id: Types.ObjectId().toString(), + }; + await AppUserProfile.deleteOne({ + userId: testUser?.id, + }); + const context = { + userId: testUser?.id, + }; + + const { acceptAdmin } = await import( + "../../../src/resolvers/Mutation/acceptAdmin" + ); + await acceptAdmin?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/addEventAttendee.spec.ts b/tests/resolvers/Mutation/addEventAttendee.spec.ts index 8c6aeb256e..621bdb9bb3 100644 --- a/tests/resolvers/Mutation/addEventAttendee.spec.ts +++ b/tests/resolvers/Mutation/addEventAttendee.spec.ts @@ -1,18 +1,18 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { EventAttendee, User } from "../../../src/models"; -import type { MutationAddEventAttendeeArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { EVENT_NOT_FOUND_ERROR, + USER_ALREADY_REGISTERED_FOR_EVENT, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, - USER_ALREADY_REGISTERED_FOR_EVENT, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { createTestUser, type TestUserType } from "../../helpers/userAndOrg"; +import { AppUserProfile, EventAttendee, User } from "../../../src/models"; +import type { MutationAddEventAttendeeArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import { createTestEvent, type TestEventType } from "../../helpers/events"; +import { createTestUser, type TestUserType } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; @@ -52,8 +52,8 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { ); await addEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -75,15 +75,15 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { }, }; - const context = { userId: randomTestUser!._id }; + const context = { userId: randomTestUser?._id }; const { addEventAttendee: addEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/addEventAttendee" ); await addEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${EVENT_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); @@ -101,19 +101,19 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { const args: MutationAddEventAttendeeArgs = { data: { userId: Types.ObjectId().toString(), - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: randomTestUser!._id }; + const context = { userId: randomTestUser?._id }; const { addEventAttendee: addEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/addEventAttendee" ); await addEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); @@ -131,19 +131,19 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { const args: MutationAddEventAttendeeArgs = { data: { userId: Types.ObjectId().toString(), - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { addEventAttendee: addEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/addEventAttendee" ); await addEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -153,12 +153,12 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { it(`registers the request user for the event successfully and returns the request user`, async () => { const args: MutationAddEventAttendeeArgs = { data: { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { addEventAttendee: addEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/addEventAttendee" @@ -167,7 +167,7 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { const payload = await addEventAttendeeResolver?.({}, args, context); const requestUser = await User.findOne({ - _id: testUser!._id, + _id: testUser?._id, }).lean(); const isUserRegistered = await EventAttendee.exists({ @@ -188,20 +188,20 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { try { const args: MutationAddEventAttendeeArgs = { data: { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { addEventAttendee: addEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/addEventAttendee" ); await addEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_ALREADY_REGISTERED_FOR_EVENT.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -209,4 +209,32 @@ describe("resolvers -> Mutation -> addEventAttendee", () => { ); } }); + it("throws an error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const args: MutationAddEventAttendeeArgs = { + data: { + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", + }, + }; + const context = { userId: testUser?._id }; + try { + const { addEventAttendee: addEventAttendeeResolver } = await import( + "../../../src/resolvers/Mutation/addEventAttendee" + ); + await addEventAttendeeResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/addFeedback.spec.ts b/tests/resolvers/Mutation/addFeedback.spec.ts index 8382cfa207..abce8e4866 100644 --- a/tests/resolvers/Mutation/addFeedback.spec.ts +++ b/tests/resolvers/Mutation/addFeedback.spec.ts @@ -1,18 +1,18 @@ import "dotenv/config"; -import { connect, disconnect } from "../../helpers/db"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import type { MutationAddFeedbackArgs } from "../../../src/types/generatedGraphQLTypes"; import type mongoose from "mongoose"; import { Types } from "mongoose"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; +import { CheckIn, EventAttendee } from "../../../src/models"; +import type { MutationAddFeedbackArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; +import { createTestEvent, type TestEventType } from "../../helpers/events"; import { EVENT_NOT_FOUND_ERROR, - USER_NOT_CHECKED_IN, FEEDBACK_ALREADY_SUBMITTED, + USER_NOT_CHECKED_IN, USER_NOT_REGISTERED_FOR_EVENT, } from "./../../../src/constants"; -import { type TestUserType, createTestUser } from "./../../helpers/userAndOrg"; -import { createTestEvent, type TestEventType } from "../../helpers/events"; -import { CheckIn, EventAttendee } from "../../../src/models"; +import { createTestUser, type TestUserType } from "./../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let randomTestUser: TestUserType; @@ -48,7 +48,7 @@ describe("resolvers -> Query -> addFeedback", () => { }; const context = { - userId: randomTestUser!._id, + userId: randomTestUser?._id, }; const { addFeedback: addFeedbackResolver } = await import( @@ -56,8 +56,8 @@ describe("resolvers -> Query -> addFeedback", () => { ); await addFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${EVENT_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); @@ -74,14 +74,14 @@ describe("resolvers -> Query -> addFeedback", () => { try { const args: MutationAddFeedbackArgs = { data: { - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", rating: 4, review: "Test Review", }, }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { addFeedback: addFeedbackResolver } = await import( @@ -89,8 +89,8 @@ describe("resolvers -> Query -> addFeedback", () => { ); await addFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_REGISTERED_FOR_EVENT.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -101,8 +101,8 @@ describe("resolvers -> Query -> addFeedback", () => { it(`throws Error if the user is has not checked in to the event`, async () => { const eventAttendee = await EventAttendee.create({ - eventId: testEvent!._id, - userId: testUser!._id, + eventId: testEvent?._id, + userId: testUser?._id, }); eventAttendeeId = eventAttendee._id; @@ -115,22 +115,22 @@ describe("resolvers -> Query -> addFeedback", () => { try { const args: MutationAddFeedbackArgs = { data: { - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", rating: 4, review: "Test Review", }, }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { addFeedback: addFeedbackResolver } = await import( "../../../src/resolvers/Mutation/addFeedback" ); await addFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_CHECKED_IN.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_CHECKED_IN.MESSAGE); @@ -148,14 +148,14 @@ describe("resolvers -> Query -> addFeedback", () => { const args: MutationAddFeedbackArgs = { data: { - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", rating: 4, review: "Test Review", }, }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { addFeedback: addFeedbackResolver } = await import( @@ -165,7 +165,7 @@ describe("resolvers -> Query -> addFeedback", () => { const payload = await addFeedbackResolver?.({}, args, context); expect(payload).toMatchObject({ - eventId: testEvent!._id, + eventId: testEvent?._id, rating: 4, review: "Test Review", }); @@ -181,22 +181,22 @@ describe("resolvers -> Query -> addFeedback", () => { try { const args: MutationAddFeedbackArgs = { data: { - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", rating: 4, review: "Test Review", }, }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { addFeedback: addFeedbackResolver } = await import( "../../../src/resolvers/Mutation/addFeedback" ); await addFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${FEEDBACK_ALREADY_SUBMITTED.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(FEEDBACK_ALREADY_SUBMITTED.MESSAGE); diff --git a/tests/resolvers/Mutation/addOrganizationCustomField.spec.ts b/tests/resolvers/Mutation/addOrganizationCustomField.spec.ts index dc89dcba37..8eb89763b2 100644 --- a/tests/resolvers/Mutation/addOrganizationCustomField.spec.ts +++ b/tests/resolvers/Mutation/addOrganizationCustomField.spec.ts @@ -1,14 +1,14 @@ -import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; -import { Types } from "mongoose"; import type mongoose from "mongoose"; +import { Types } from "mongoose"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { addOrganizationCustomField } from "../../../src/resolvers/Mutation/addOrganizationCustomField"; +import { connect, disconnect } from "../../helpers/db"; import { createTestUser, createTestUserAndOrganization, type TestOrganizationType, type TestUserType, } from "../../helpers/userAndOrg"; -import { connect, disconnect } from "../../helpers/db"; import { CUSTOM_FIELD_NAME_MISSING, @@ -17,6 +17,7 @@ import { USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; +import { AppUserProfile } from "../../../src/models"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -74,9 +75,9 @@ describe("resolvers => Mutation => addOrganizationCustomField", () => { try { await addOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); } @@ -98,9 +99,9 @@ describe("resolvers => Mutation => addOrganizationCustomField", () => { try { await addOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(CUSTOM_FIELD_NAME_MISSING.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${CUSTOM_FIELD_NAME_MISSING.MESSAGE}` ); } @@ -122,9 +123,9 @@ describe("resolvers => Mutation => addOrganizationCustomField", () => { try { await addOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(CUSTOM_FIELD_TYPE_MISSING.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${CUSTOM_FIELD_TYPE_MISSING.MESSAGE}` ); } @@ -150,9 +151,9 @@ describe("resolvers => Mutation => addOrganizationCustomField", () => { try { await addOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -173,13 +174,40 @@ describe("resolvers => Mutation => addOrganizationCustomField", () => { try { await addOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith( ORGANIZATION_NOT_FOUND_ERROR.MESSAGE ); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${ORGANIZATION_NOT_FOUND_ERROR.MESSAGE}` ); } }); + it("throws an error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + + const args = { + organizationId: testOrganization?._id, + name: "testName", + type: "testType", + }; + const testUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const context = { userId: testUser?._id }; + + try { + await addOrganizationCustomField?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/addUserImage.spec.ts b/tests/resolvers/Mutation/addUserImage.spec.ts index 889dee4483..d7519d2552 100644 --- a/tests/resolvers/Mutation/addUserImage.spec.ts +++ b/tests/resolvers/Mutation/addUserImage.spec.ts @@ -3,21 +3,21 @@ import type mongoose from "mongoose"; import { Types } from "mongoose"; import { connect, disconnect } from "../../helpers/db"; -import type { MutationAddUserImageArgs } from "../../../src/types/generatedGraphQLTypes"; -import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; +import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; +import { addUserImage as addUserImageResolverUserImage } from "../../../src/resolvers/Mutation/addUserImage"; +import type { MutationAddUserImageArgs } from "../../../src/types/generatedGraphQLTypes"; +import * as uploadEncodedImage from "../../../src/utilities/encodedImageStorage/uploadEncodedImage"; import type { TestUserType } from "../../helpers/userAndOrg"; import { createTestUser } from "../../helpers/userAndOrg"; -import * as uploadEncodedImage from "../../../src/utilities/encodedImageStorage/uploadEncodedImage"; -import { addUserImage as addUserImageResolverUserImage } from "../../../src/resolvers/Mutation/addUserImage"; let testUser: TestUserType; let MONGOOSE_INSTANCE: typeof mongoose; @@ -62,9 +62,9 @@ describe("resolvers -> Mutation -> addUserImage", () => { ); await addUserImageResolverUserError?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -86,7 +86,9 @@ describe("resolvers -> Mutation -> addUserImage", () => { args, context ); - + // const testApp = await AppUserProfile.findOne({ userId: testUser?._id }); + // console.log(testApp); + // console.log(addUserImagePayload, testUser?.appUserProfileId); expect(addUserImagePayload).toEqual({ ...testUser?.toObject(), createdAt: addUserImagePayload?.createdAt, diff --git a/tests/resolvers/Mutation/adminRemoveEvent.spec.ts b/tests/resolvers/Mutation/adminRemoveEvent.spec.ts index 863f77d0ed..dd9f283663 100644 --- a/tests/resolvers/Mutation/adminRemoveEvent.spec.ts +++ b/tests/resolvers/Mutation/adminRemoveEvent.spec.ts @@ -1,26 +1,28 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Organization, Event } from "../../../src/models"; +import { AppUserProfile, Event, Organization, User } from "../../../src/models"; import type { MutationAdminRemoveEventArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { adminRemoveEvent as adminRemoveEventResolver } from "../../../src/resolvers/Mutation/adminRemoveEvent"; +import { nanoid } from "nanoid"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { EVENT_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ADMIN, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; +import { adminRemoveEvent as adminRemoveEventResolver } from "../../../src/resolvers/Mutation/adminRemoveEvent"; +import { cacheEvents } from "../../../src/services/EventCache/cacheEvents"; +import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; +import type { TestEventType } from "../../helpers/events"; +import { createTestEvent } from "../../helpers/events"; import type { - TestUserType, TestOrganizationType, + TestUserType, } from "../../helpers/userAndOrg"; -import type { TestEventType } from "../../helpers/events"; -import { createTestEvent } from "../../helpers/events"; -import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; -import { cacheEvents } from "../../../src/services/EventCache/cacheEvents"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -60,8 +62,8 @@ describe("resolvers -> Mutation -> adminRemoveEvent", () => { }; await adminRemoveEventResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -91,8 +93,10 @@ describe("resolvers -> Mutation -> adminRemoveEvent", () => { }; await adminRemoveEventResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -124,8 +128,8 @@ describe("resolvers -> Mutation -> adminRemoveEvent", () => { }; await adminRemoveEventResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -159,11 +163,37 @@ describe("resolvers -> Mutation -> adminRemoveEvent", () => { }; await adminRemoveEventResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); + it("throws an error if the user does not have appUserProfile", async () => { + try { + const args: MutationAdminRemoveEventArgs = { + eventId: testEvent?.id, + }; + + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + + const context = { + userId: newUser.id, + }; + await adminRemoveEventResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); it(`removes event with _id === args.eventId and returns it`, async () => { const updatedOrganization = await Organization.findOneAndUpdate( { @@ -205,14 +235,23 @@ describe("resolvers -> Mutation -> adminRemoveEvent", () => { const testUpdatedUser = await User.findOne({ _id: testUser?._id, }) - .select(["createdEvents", "eventAdmin", "registeredEvents"]) + .select(["registeredEvents"]) .lean(); + const testUpdatedAppUser = await AppUserProfile.findOne({ + userId: testUser?._id, + }) + .select(["createdEvents", "eventAdmin"]) + .lean(); expect(testUpdatedUser).toEqual( + expect.objectContaining({ + registeredEvents: expect.arrayContaining([]), + }) + ); + expect(testUpdatedAppUser).toEqual( expect.objectContaining({ createdEvents: expect.arrayContaining([]), eventAdmin: expect.arrayContaining([]), - registeredEvents: expect.arrayContaining([]), }) ); }); diff --git a/tests/resolvers/Mutation/adminRemoveGroup.spec.ts b/tests/resolvers/Mutation/adminRemoveGroup.spec.ts index 6e6610c06d..fea1f422df 100644 --- a/tests/resolvers/Mutation/adminRemoveGroup.spec.ts +++ b/tests/resolvers/Mutation/adminRemoveGroup.spec.ts @@ -1,25 +1,27 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { Organization, GroupChat } from "../../../src/models"; +import { GroupChat, Organization, User } from "../../../src/models"; import type { MutationAdminRemoveGroupArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { adminRemoveGroup as adminRemoveGroupResolver } from "../../../src/resolvers/Mutation/adminRemoveGroup"; +import { nanoid } from "nanoid"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { CHAT_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ADMIN, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; +import { adminRemoveGroup as adminRemoveGroupResolver } from "../../../src/resolvers/Mutation/adminRemoveGroup"; +import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; +import type { TestGroupChatType } from "../../helpers/groupChat"; +import { createTestGroupChat } from "../../helpers/groupChat"; import type { - TestUserType, TestOrganizationType, + TestUserType, } from "../../helpers/userAndOrg"; -import type { TestGroupChatType } from "../../helpers/groupChat"; -import { createTestGroupChat } from "../../helpers/groupChat"; -import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -44,6 +46,31 @@ afterAll(async () => { }); describe("resolvers -> Mutation -> adminRemoveGroup", () => { + it("throws an error if the user does not have appUserProfile", async () => { + try { + const args: MutationAdminRemoveGroupArgs = { + groupId: testGroupChat?.id, + }; + + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + + const context = { + userId: newUser?.id, + }; + await adminRemoveGroupResolver?.({}, args, context); + } catch (error: unknown) { + // console.log(error);? + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); it(`throws NotFoundError if no groupChat exists with _id === args.groupId`, async () => { try { const args: MutationAdminRemoveGroupArgs = { @@ -55,8 +82,8 @@ describe("resolvers -> Mutation -> adminRemoveGroup", () => { }; await adminRemoveGroupResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(CHAT_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(CHAT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -83,8 +110,10 @@ describe("resolvers -> Mutation -> adminRemoveGroup", () => { }; await adminRemoveGroupResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -110,8 +139,8 @@ describe("resolvers -> Mutation -> adminRemoveGroup", () => { }; await adminRemoveGroupResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -142,8 +171,7 @@ describe("resolvers -> Mutation -> adminRemoveGroup", () => { new: true, } ); - - cacheOrganizations([updatedOrganization!]); + if (updatedOrganization) cacheOrganizations([updatedOrganization]); const args: MutationAdminRemoveGroupArgs = { groupId: testGroupChat?.id, @@ -154,8 +182,10 @@ describe("resolvers -> Mutation -> adminRemoveGroup", () => { }; await adminRemoveGroupResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); @@ -173,8 +203,7 @@ describe("resolvers -> Mutation -> adminRemoveGroup", () => { new: true, } ); - - cacheOrganizations([updatedOrganization!]); + if (updatedOrganization) cacheOrganizations([updatedOrganization]); const args: MutationAdminRemoveGroupArgs = { groupId: testGroupChat?.id, diff --git a/tests/resolvers/Mutation/assignUserTag.spec.ts b/tests/resolvers/Mutation/assignUserTag.spec.ts index f9eda54f3f..34d8e4b0c8 100644 --- a/tests/resolvers/Mutation/assignUserTag.spec.ts +++ b/tests/resolvers/Mutation/assignUserTag.spec.ts @@ -5,26 +5,26 @@ import type { MutationAssignUserTagArgs } from "../../../src/types/generatedGrap import { connect, disconnect } from "../../helpers/db"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, - TAG_NOT_FOUND, - USER_DOES_NOT_BELONG_TO_TAGS_ORGANIZATION, - USER_ALREADY_HAS_TAG, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; -import { createTestUser } from "../../helpers/userAndOrg"; +import { + TAG_NOT_FOUND, + USER_ALREADY_HAS_TAG, + USER_DOES_NOT_BELONG_TO_TAGS_ORGANIZATION, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile, TagUser } from "../../../src/models"; import type { TestUserTagType } from "../../helpers/tags"; import { createRootTagWithOrg } from "../../helpers/tags"; -import { TagUser } from "../../../src/models"; +import type { TestUserType } from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -71,8 +71,8 @@ describe("resolvers -> Mutation -> assignUserTag", () => { ); await assignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -101,8 +101,8 @@ describe("resolvers -> Mutation -> assignUserTag", () => { ); await assignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -133,9 +133,11 @@ describe("resolvers -> Mutation -> assignUserTag", () => { ); await assignUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(TAG_NOT_FOUND.MESSAGE); - expect(error.message).toEqual(`Translated ${TAG_NOT_FOUND.MESSAGE}`); + expect((error as Error).message).toEqual( + `Translated ${TAG_NOT_FOUND.MESSAGE}` + ); } }); @@ -163,8 +165,8 @@ describe("resolvers -> Mutation -> assignUserTag", () => { ); await assignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -197,8 +199,8 @@ describe("resolvers -> Mutation -> assignUserTag", () => { ); await assignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_DOES_NOT_BELONG_TO_TAGS_ORGANIZATION.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -256,11 +258,46 @@ describe("resolvers -> Mutation -> assignUserTag", () => { ); await assignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_ALREADY_HAS_TAG.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(`${USER_ALREADY_HAS_TAG.MESSAGE}`); } }); + it("throws error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + + try { + const args: MutationAssignUserTagArgs = { + input: { + userId: randomUser?._id, + tagId: testTag?._id.toString() ?? "", + }, + }; + const temp = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: temp?._id, + }); + const context = { + userId: temp?._id, + }; + + const { assignUserTag: assignUserTagResolver } = await import( + "../../../src/resolvers/Mutation/assignUserTag" + ); + + await assignUserTagResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + expect(spy).toHaveBeenLastCalledWith( + `${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/blockPluginCreationBySuperadmin.spec.ts b/tests/resolvers/Mutation/blockPluginCreationBySuperadmin.spec.ts index 903ca5a3c4..3dd1cfe40a 100644 --- a/tests/resolvers/Mutation/blockPluginCreationBySuperadmin.spec.ts +++ b/tests/resolvers/Mutation/blockPluginCreationBySuperadmin.spec.ts @@ -1,24 +1,26 @@ import "dotenv/config"; -import { User } from "../../../src/models"; import type mongoose from "mongoose"; import { Types } from "mongoose"; +import { AppUserProfile, User } from "../../../src/models"; import type { MutationBlockPluginCreationBySuperadminArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { blockPluginCreationBySuperadmin as blockPluginCreationBySuperadminResolver } from "../../../src/resolvers/Mutation/blockPluginCreationBySuperadmin"; -import { - USER_NOT_AUTHORIZED_SUPERADMIN, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; +import { nanoid } from "nanoid"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; +import { + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { blockPluginCreationBySuperadmin as blockPluginCreationBySuperadminResolver } from "../../../src/resolvers/Mutation/blockPluginCreationBySuperadmin"; import type { TestUserType } from "../../helpers/userAndOrg"; import { createTestUser } from "../../helpers/userAndOrg"; @@ -56,8 +58,70 @@ describe("resolvers -> Mutation -> blockPluginCreationBySuperadmin", () => { }; await blockPluginCreationBySuperadminResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } + }); + it("throws error if user does not have AppUserProfile", async () => { + try { + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const args: MutationBlockPluginCreationBySuperadminArgs = { + blockUser: false, + userId: newUser?.id, + }; + await AppUserProfile.updateOne( + { + userId: testUser?._id, + }, + { + isSuperAdmin: true, + } + ); + + const context = { + userId: testUser?.id, + }; + + await blockPluginCreationBySuperadminResolver?.({}, args, context); + } catch (error: unknown) { + // console.log((error as Error).message); + + expect((error as Error).message).toEqual( + `${USER_NOT_FOUND_ERROR.MESSAGE}` + ); + } + }); + it("throws error if current appUser does not have AppUserProfile", async () => { + try { + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const args: MutationBlockPluginCreationBySuperadminArgs = { + blockUser: false, + userId: testUser?.id, + }; + + const context = { + userId: newUser?.id, + }; + + await blockPluginCreationBySuperadminResolver?.({}, args, context); + } catch (error: unknown) { + // console.log((error as Error).message); + + expect((error as Error).message).toEqual( + `${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); } }); @@ -77,6 +141,7 @@ describe("resolvers -> Mutation -> blockPluginCreationBySuperadmin", () => { const context = { userId: testUser?.id, }; + // console.log(testUser) const { blockPluginCreationBySuperadmin: blockPluginCreationBySuperadminError, @@ -85,9 +150,10 @@ describe("resolvers -> Mutation -> blockPluginCreationBySuperadmin", () => { ); await blockPluginCreationBySuperadminError?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { + // console.log(`-----------------${(error as Error).message}`); expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE}` ); } @@ -95,12 +161,12 @@ describe("resolvers -> Mutation -> blockPluginCreationBySuperadmin", () => { it(`depending on args.blockUser blocks/unblocks plugin creation for user with _id === args.userId and returns the user`, async () => { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, } ); @@ -116,8 +182,8 @@ describe("resolvers -> Mutation -> blockPluginCreationBySuperadmin", () => { const blockPluginCreationBySuperadminPayload = await blockPluginCreationBySuperadminResolver?.({}, args, context); - const testUpdatedTestUser = await User.findOne({ - _id: testUser?.id, + const testUpdatedTestUser = await AppUserProfile.findOne({ + userId: testUser?._id, }).lean(); expect(blockPluginCreationBySuperadminPayload).toEqual(testUpdatedTestUser); diff --git a/tests/resolvers/Mutation/blockUser.spec.ts b/tests/resolvers/Mutation/blockUser.spec.ts index d63096aa1d..a0897ea16b 100644 --- a/tests/resolvers/Mutation/blockUser.spec.ts +++ b/tests/resolvers/Mutation/blockUser.spec.ts @@ -1,11 +1,20 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Organization } from "../../../src/models"; +import { AppUserProfile, Organization, User } from "../../../src/models"; import type { MutationBlockUserArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { blockUser as blockUserResolver } from "../../../src/resolvers/Mutation/blockUser"; +import { nanoid } from "nanoid"; +import { + afterAll, + afterEach, + beforeAll, + describe, + expect, + it, + vi, +} from "vitest"; import { MEMBER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, @@ -14,22 +23,13 @@ import { USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { nanoid } from "nanoid"; -import { - beforeAll, - afterAll, - describe, - it, - expect, - vi, - afterEach, -} from "vitest"; +import { blockUser as blockUserResolver } from "../../../src/resolvers/Mutation/blockUser"; +import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; import type { - TestUserType, TestOrganizationType, + TestUserType, } from "../../helpers/userAndOrg"; import { createTestUser } from "../../helpers/userAndOrg"; -import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; let testUser: TestUserType; let testUser2: TestUserType; @@ -58,11 +58,20 @@ beforeAll(async () => { }, { $set: { - createdOrganizations: [testOrganization._id], joinedOrganizations: [testOrganization._id], }, } ); + await AppUserProfile.updateOne( + { + userId: testUser?._id, + }, + { + $set: { + createdOrganizations: [testOrganization._id], + }, + } + ); const { requestContext } = await import("../../../src/libraries"); vi.spyOn(requestContext, "translate").mockImplementation( (message) => message @@ -90,8 +99,10 @@ describe("resolvers -> Mutation -> blockUser", () => { }; await blockUserResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -107,8 +118,8 @@ describe("resolvers -> Mutation -> blockUser", () => { }; await blockUserResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -132,8 +143,8 @@ describe("resolvers -> Mutation -> blockUser", () => { ); await blockUserResolverError?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(MEMBER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(MEMBER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -157,8 +168,8 @@ describe("resolvers -> Mutation -> blockUser", () => { ); await blockUserResolverError?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_BLOCKING_SELF.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_BLOCKING_SELF.MESSAGE); } }); @@ -193,8 +204,10 @@ describe("resolvers -> Mutation -> blockUser", () => { }; await blockUserResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); @@ -220,9 +233,9 @@ describe("resolvers -> Mutation -> blockUser", () => { await cacheOrganizations([updatedOrganization]); } - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?.id, + userId: testUser?.id, }, { $push: { @@ -241,8 +254,10 @@ describe("resolvers -> Mutation -> blockUser", () => { }; await blockUserResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); diff --git a/tests/resolvers/Mutation/checkIn.spec.ts b/tests/resolvers/Mutation/checkIn.spec.ts index 373c16dd76..b7b99becf2 100644 --- a/tests/resolvers/Mutation/checkIn.spec.ts +++ b/tests/resolvers/Mutation/checkIn.spec.ts @@ -1,20 +1,20 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { EventAttendee } from "../../../src/models"; -import type { MutationCheckInArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { EVENT_NOT_FOUND_ERROR, + USER_ALREADY_CHECKED_IN, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, USER_NOT_REGISTERED_FOR_EVENT, - USER_ALREADY_CHECKED_IN, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { createTestUser, type TestUserType } from "../../helpers/userAndOrg"; +import { AppUserProfile, EventAttendee } from "../../../src/models"; +import type { MutationCheckInArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import { type TestEventType } from "../../helpers/events"; import { createTestEventWithRegistrants } from "../../helpers/eventsWithRegistrants"; +import { createTestUser, type TestUserType } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; @@ -54,8 +54,8 @@ describe("resolvers -> Mutation -> checkIn", () => { ); await checkInResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -77,15 +77,15 @@ describe("resolvers -> Mutation -> checkIn", () => { }, }; - const context = { userId: randomTestUser!._id }; + const context = { userId: randomTestUser?._id }; const { checkIn: checkInResolver } = await import( "../../../src/resolvers/Mutation/checkIn" ); await checkInResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${EVENT_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); @@ -103,19 +103,19 @@ describe("resolvers -> Mutation -> checkIn", () => { const args: MutationCheckInArgs = { data: { userId: Types.ObjectId().toString(), - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: randomTestUser!._id }; + const context = { userId: randomTestUser?._id }; const { checkIn: checkInResolver } = await import( "../../../src/resolvers/Mutation/checkIn" ); await checkInResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); @@ -133,19 +133,19 @@ describe("resolvers -> Mutation -> checkIn", () => { const args: MutationCheckInArgs = { data: { userId: Types.ObjectId().toString(), - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { checkIn: checkInResolver } = await import( "../../../src/resolvers/Mutation/checkIn" ); await checkInResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -162,20 +162,20 @@ describe("resolvers -> Mutation -> checkIn", () => { try { const args: MutationCheckInArgs = { data: { - userId: randomTestUser!._id, - eventId: testEvent!._id, + userId: randomTestUser?._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { checkIn: checkInResolver } = await import( "../../../src/resolvers/Mutation/checkIn" ); await checkInResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_REGISTERED_FOR_EVENT.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -187,14 +187,14 @@ describe("resolvers -> Mutation -> checkIn", () => { it(`Checks the user in successfully`, async () => { const args: MutationCheckInArgs = { data: { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", allotedRoom: "test room", allotedSeat: "test seat", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { checkIn: checkInResolver } = await import( "../../../src/resolvers/Mutation/checkIn" @@ -203,13 +203,13 @@ describe("resolvers -> Mutation -> checkIn", () => { const payload = await checkInResolver?.({}, args, context); const eventAttendee = await EventAttendee.findOne({ - eventId: testEvent!._id, - userId: testUser!._id, + eventId: testEvent?._id, + userId: testUser?._id, }).lean(); - expect(eventAttendee!.checkInId).not.toBeNull(); + expect(eventAttendee?.checkInId).not.toBeNull(); expect(payload).toMatchObject({ - eventAttendeeId: eventAttendee!._id, + eventAttendeeId: eventAttendee?._id, allotedSeat: "test seat", allotedRoom: "test room", }); @@ -225,23 +225,50 @@ describe("resolvers -> Mutation -> checkIn", () => { try { const args: MutationCheckInArgs = { data: { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { checkIn: checkInResolver } = await import( "../../../src/resolvers/Mutation/checkIn" ); await checkInResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_ALREADY_CHECKED_IN.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_ALREADY_CHECKED_IN.MESSAGE); } }); + it("throws an error if user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + const args: MutationCheckInArgs = { + data: { + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", + }, + }; + const context = { userId: testUser?._id }; + const { checkIn: checkInResolver } = await import( + "../../../src/resolvers/Mutation/checkIn" + ); + try { + await checkInResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/createActionItem.spec.ts b/tests/resolvers/Mutation/createActionItem.spec.ts index c2b0ec22fc..42afd4c23b 100644 --- a/tests/resolvers/Mutation/createActionItem.spec.ts +++ b/tests/resolvers/Mutation/createActionItem.spec.ts @@ -1,32 +1,32 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import type { MutationCreateActionItemArgs } from "../../../src/types/generatedGraphQLTypes"; -import { createActionItem as createActionItemResolver } from "../../../src/resolvers/Mutation/createActionItem"; -import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { - USER_NOT_FOUND_ERROR, ACTION_ITEM_CATEGORY_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, EVENT_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, USER_NOT_MEMBER_FOR_ORGANIZATION, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { createTestUser } from "../../helpers/userAndOrg"; +import { createActionItem as createActionItemResolver } from "../../../src/resolvers/Mutation/createActionItem"; +import type { MutationCreateActionItemArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; +import { nanoid } from "nanoid"; +import { AppUserProfile, Event, User } from "../../../src/models"; import type { TestActionItemCategoryType } from "../../helpers/actionItemCategory"; import { createTestCategory } from "../../helpers/actionItemCategory"; import type { TestEventType } from "../../helpers/events"; -import { Event, User } from "../../../src/models"; -import { nanoid } from "nanoid"; let randomUser: TestUserType; let randomUser2: TestUserType; -let superAdminTestUser: TestUserType; +// let superAdminTestUserAppProfile: TestAppUserProfileType; let testUser: TestUserType; let testOrganization: TestOrganizationType; let testCategory: TestActionItemCategoryType; @@ -43,15 +43,12 @@ beforeAll(async () => { randomUser = await createTestUser(); randomUser2 = await createTestUser(); - superAdminTestUser = await User.findOneAndUpdate( - { - _id: randomUser2?._id, - }, + await AppUserProfile.updateOne( { - userType: "SUPERADMIN", + userId: randomUser2?._id, }, { - new: true, + isSuperAdmin: true, } ); @@ -90,8 +87,8 @@ describe("resolvers -> Mutation -> createActionItem", () => { }; await createActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -109,8 +106,8 @@ describe("resolvers -> Mutation -> createActionItem", () => { }; await createActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( ACTION_ITEM_CATEGORY_NOT_FOUND_ERROR.MESSAGE ); } @@ -130,8 +127,8 @@ describe("resolvers -> Mutation -> createActionItem", () => { }; await createActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -149,8 +146,10 @@ describe("resolvers -> Mutation -> createActionItem", () => { }; await createActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_MEMBER_FOR_ORGANIZATION.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_MEMBER_FOR_ORGANIZATION.MESSAGE + ); } }); @@ -178,8 +177,8 @@ describe("resolvers -> Mutation -> createActionItem", () => { }; await createActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -197,8 +196,10 @@ describe("resolvers -> Mutation -> createActionItem", () => { }; await createActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); @@ -206,7 +207,7 @@ describe("resolvers -> Mutation -> createActionItem", () => { const args: MutationCreateActionItemArgs = { data: { assigneeId: randomUser?._id, - eventId: testEvent?._id, + eventId: testEvent?._id.toString() ?? "", }, actionItemCategoryId: testCategory?._id, }; @@ -262,8 +263,12 @@ describe("resolvers -> Mutation -> createActionItem", () => { }; const context = { - userId: superAdminTestUser?._id, + userId: randomUser2?._id, }; + // const superAdmin = await AppUserProfile.findOne({ + // userId: randomUser2?._id, + // }); + // console.log(superAdmin) const createActionItemPayload = await createActionItemResolver?.( {}, @@ -277,4 +282,25 @@ describe("resolvers -> Mutation -> createActionItem", () => { }) ); }); + it("throws error if the user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: randomUser?._id, + }); + const args: MutationCreateActionItemArgs = { + data: { + assigneeId: randomUser?._id, + }, + actionItemCategoryId: testCategory?._id, + }; + const context = { + userId: randomUser?._id, + }; + try { + await createActionItemResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/createActionItemCategory.spec.ts b/tests/resolvers/Mutation/createActionItemCategory.spec.ts index f796341472..3894f20a0e 100644 --- a/tests/resolvers/Mutation/createActionItemCategory.spec.ts +++ b/tests/resolvers/Mutation/createActionItemCategory.spec.ts @@ -1,26 +1,26 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import type { MutationCreateActionItemCategoryArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; -import { createActionItemCategory as createActionItemCategoryResolver } from "../../../src/resolvers/Mutation/createActionItemCategory"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { + ACTION_ITEM_CATEGORY_ALREADY_EXISTS, ORGANIZATION_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ADMIN, - ACTION_ITEM_CATEGORY_ALREADY_EXISTS, + USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { - createTestUser, - createTestUserAndOrganization, -} from "../../helpers/userAndOrg"; +import { createActionItemCategory as createActionItemCategoryResolver } from "../../../src/resolvers/Mutation/createActionItemCategory"; +import type { MutationCreateActionItemCategoryArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; +import { + createTestUser, + createTestUserAndOrganization, +} from "../../helpers/userAndOrg"; -import { Organization, User } from "../../../src/models"; +import { AppUserProfile } from "../../../src/models"; let randomUser: TestUserType; let testUser: TestUserType; @@ -56,8 +56,8 @@ describe("resolvers -> Mutation -> createCategory", () => { }; await createActionItemCategoryResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -73,8 +73,10 @@ describe("resolvers -> Mutation -> createCategory", () => { }; await createActionItemCategoryResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -90,8 +92,10 @@ describe("resolvers -> Mutation -> createCategory", () => { }; await createActionItemCategoryResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); @@ -120,12 +124,12 @@ describe("resolvers -> Mutation -> createCategory", () => { }); it(`creates the actionItemCategory and returns it as superAdmin`, async () => { - const superAdminTestUser = await User.findOneAndUpdate( + const superAdminTestUser = await AppUserProfile.findOneAndUpdate( { - _id: randomUser?._id, + userId: randomUser?._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, }, { new: true, @@ -138,7 +142,7 @@ describe("resolvers -> Mutation -> createCategory", () => { }; const context = { - userId: superAdminTestUser?._id, + userId: superAdminTestUser?.userId, }; const createCategoryPayload = await createActionItemCategoryResolver?.( @@ -167,8 +171,8 @@ describe("resolvers -> Mutation -> createCategory", () => { }; await createActionItemCategoryResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( ACTION_ITEM_CATEGORY_ALREADY_EXISTS.MESSAGE ); } diff --git a/tests/resolvers/Mutation/createAdmin.spec.ts b/tests/resolvers/Mutation/createAdmin.spec.ts index 7b2706ca4c..31349562b5 100644 --- a/tests/resolvers/Mutation/createAdmin.spec.ts +++ b/tests/resolvers/Mutation/createAdmin.spec.ts @@ -1,24 +1,25 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Organization } from "../../../src/models"; +import { AppUserProfile, Organization, User } from "../../../src/models"; import type { MutationCreateAdminArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { createAdmin as createAdminResolver } from "../../../src/resolvers/Mutation/createAdmin"; +import { nanoid } from "nanoid"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { ORGANIZATION_MEMBER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; +import { createAdmin as createAdminResolver } from "../../../src/resolvers/Mutation/createAdmin"; +import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; import { createTestUserAndOrganization } from "../../helpers/userAndOrg"; -import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -55,8 +56,10 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createAdminResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -85,8 +88,8 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createAdminResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -103,13 +106,13 @@ describe("resolvers -> Mutation -> createAdmin", () => { } ); - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $set: { - userType: "SUPERADMIN", + isSuperAdmin: true, }, } ); @@ -126,8 +129,59 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createAdminResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } + }); + it("throws error if user does not have AppUserProfile", async () => { + try { + const args: MutationCreateAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: Types.ObjectId().toString(), + }, + }; + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const context = { + userId: newUser?.id, + }; + await createAdminResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); + it("throws error if user does not exists", async () => { + try { + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const args: MutationCreateAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: newUser.id, + }, + }; + + const context = { + userId: testUser?.id, + }; + await createAdminResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `${USER_NOT_FOUND_ERROR.MESSAGE}` + ); } }); @@ -146,8 +200,8 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createAdminResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( ORGANIZATION_MEMBER_NOT_FOUND_ERROR.MESSAGE ); } @@ -186,8 +240,10 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createAdminResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); @@ -223,8 +279,8 @@ describe("resolvers -> Mutation -> createAdmin", () => { const createAdminPayload = await createAdminResolver?.({}, args, context); - const updatedTestUser = await User.findOne({ - _id: testUser?._id, + const updatedTestUser = await AppUserProfile.findOne({ + userId: testUser?._id, }) .select(["-password"]) .lean(); @@ -239,4 +295,24 @@ describe("resolvers -> Mutation -> createAdmin", () => { expect(updatedTestOrganization?.admins).toEqual([testUser?._id]); }); + it("throws error if user does not exists", async () => { + try { + const args: MutationCreateAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: Types.ObjectId().toString(), + }, + }; + + const context = { + userId: testUser?.id, + }; + await createAdminResolver?.({}, args, context); + } catch (error: unknown) { + // console.log(error) + expect((error as Error).message).toEqual( + `${USER_NOT_FOUND_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/createEvent.spec.ts b/tests/resolvers/Mutation/createEvent.spec.ts index 9a20ba9884..d0498402ff 100644 --- a/tests/resolvers/Mutation/createEvent.spec.ts +++ b/tests/resolvers/Mutation/createEvent.spec.ts @@ -1,29 +1,38 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Organization, EventAttendee, Event } from "../../../src/models"; +import { + AppUserProfile, + Event, + EventAttendee, + Organization, + User, +} from "../../../src/models"; import type { MutationCreateEventArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; +import { fail } from "assert"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { LENGTH_VALIDATION_ERROR, ORGANIZATION_NOT_AUTHORIZED_ERROR, ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import type { - TestUserType, - TestOrganizationType, -} from "../../helpers/userAndOrg"; -import { createTestUser } from "../../helpers/userAndOrg"; import { InputValidationError, NotFoundError, UnauthorizedError, } from "../../../src/libraries/errors"; -import { fail } from "assert"; +import type { + TestAppUserProfileType, + TestOrganizationType, + TestUserType, +} from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; let testUser: TestUserType; +let testAppUserProfile: TestAppUserProfileType; let testOrganization: TestOrganizationType; let MONGOOSE_INSTANCE: typeof mongoose; @@ -31,19 +40,20 @@ beforeAll(async () => { MONGOOSE_INSTANCE = await connect(); testUser = await createTestUser(); + testAppUserProfile = await AppUserProfile.findOne({ userId: testUser?._id }); testOrganization = await Organization.create({ name: "name", description: "description", - isPublic: true, + userRegistrationRequired: true, creatorId: testUser?._id, admins: [testUser?._id], members: [testUser?._id], visibleInSearch: true, }); - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $push: { @@ -174,11 +184,20 @@ describe("resolvers -> Mutation -> createEvent", () => { }, { $push: { - createdOrganizations: testOrganization?._id, joinedOrganizations: testOrganization?._id, }, } ); + await AppUserProfile.updateOne( + { + userId: testUser?._id, + }, + { + $push: { + createdOrganizations: testOrganization?._id, + }, + } + ); const args: MutationCreateEventArgs = { data: { @@ -244,13 +263,11 @@ describe("resolvers -> Mutation -> createEvent", () => { const updatedTestUser = await User.findOne({ _id: testUser?._id, }) - .select(["eventAdmin", "createdEvents", "registeredEvents"]) + .select(["registeredEvents"]) .lean(); expect(updatedTestUser).toEqual( expect.objectContaining({ - eventAdmin: expect.arrayContaining([createEventPayload?._id]), - createdEvents: expect.arrayContaining([createEventPayload?._id]), registeredEvents: expect.arrayContaining([createEventPayload?._id]), }) ); @@ -263,11 +280,20 @@ describe("resolvers -> Mutation -> createEvent", () => { }, { $push: { - createdOrganizations: testOrganization?._id, joinedOrganizations: testOrganization?._id, }, } ); + await AppUserProfile.updateOne( + { + _id: testAppUserProfile?._id, + }, + { + $push: { + createdOrganizations: testOrganization?._id, + }, + } + ); const args: MutationCreateEventArgs = { data: { @@ -334,13 +360,11 @@ describe("resolvers -> Mutation -> createEvent", () => { const updatedTestUser = await User.findOne({ _id: testUser?._id, }) - .select(["eventAdmin", "createdEvents", "registeredEvents"]) + .select(["registeredEvents"]) .lean(); expect(updatedTestUser).toEqual( expect.objectContaining({ - eventAdmin: expect.arrayContaining([createEventPayload?._id]), - createdEvents: expect.arrayContaining([createEventPayload?._id]), registeredEvents: expect.arrayContaining([createEventPayload?._id]), }) ); @@ -353,11 +377,20 @@ describe("resolvers -> Mutation -> createEvent", () => { }, { $push: { - createdOrganizations: testOrganization?._id, joinedOrganizations: testOrganization?._id, }, } ); + await AppUserProfile.updateOne( + { + _id: testAppUserProfile?._id, + }, + { + $push: { + createdOrganizations: testOrganization?._id, + }, + } + ); const args: MutationCreateEventArgs = { data: { @@ -423,13 +456,11 @@ describe("resolvers -> Mutation -> createEvent", () => { const updatedTestUser = await User.findOne({ _id: testUser?._id, }) - .select(["eventAdmin", "createdEvents", "registeredEvents"]) + .select(["registeredEvents"]) .lean(); expect(updatedTestUser).toEqual( expect.objectContaining({ - eventAdmin: expect.arrayContaining([createEventPayload?._id]), - createdEvents: expect.arrayContaining([createEventPayload?._id]), registeredEvents: expect.arrayContaining([createEventPayload?._id]), }) ); @@ -675,4 +706,51 @@ describe("Check for validation conditions", () => { } } }); + it("throws error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + vi.spyOn(requestContext, "translate").mockImplementation( + (message) => message + ); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + + const args: MutationCreateEventArgs = { + data: { + organizationId: testOrganization?.id, + allDay: false, + description: "Random", + endDate: "Tue Feb 15 2023", + endTime: "", + isPublic: false, + isRegisterable: false, + latitude: 1, + longitude: 1, + location: "Random", + recurring: false, + startDate: "Tue Feb 14 2023", + startTime: "", + title: "Random", + recurrance: "DAILY", + }, + }; + + const context = { + userId: testUser?.id, + }; + + const { createEvent: createEventResolverError } = await import( + "../../../src/resolvers/Mutation/createEvent" + ); + + try { + await createEventResolverError?.({}, args, context); + } catch (error: unknown) { + if (error instanceof UnauthorizedError) { + expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + } else { + fail(`Expected UnauthorizedError, but got ${error}`); + } + } + }); }); diff --git a/tests/resolvers/Mutation/createMember.spec.ts b/tests/resolvers/Mutation/createMember.spec.ts index f217990fa0..962f91155c 100644 --- a/tests/resolvers/Mutation/createMember.spec.ts +++ b/tests/resolvers/Mutation/createMember.spec.ts @@ -1,17 +1,18 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Organization } from "../../../src/models"; +import { AppUserProfile, Organization } from "../../../src/models"; import type { MutationCreateMemberArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { createMember as createMemberResolver } from "../../../src/resolvers/Mutation/createMember"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { + MEMBER_NOT_FOUND_ERROR, ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, - MEMBER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; +import { createMember as createMemberResolver } from "../../../src/resolvers/Mutation/createMember"; import type { TestOrganizationType, TestUserType, @@ -64,8 +65,8 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createMemberResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -82,13 +83,13 @@ describe("resolvers -> Mutation -> createAdmin", () => { } ); - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $set: { - userType: "SUPERADMIN", + isSuperAdmin: true, }, } ); @@ -105,8 +106,8 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createMemberResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -124,8 +125,10 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createMemberResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -155,8 +158,8 @@ describe("resolvers -> Mutation -> createAdmin", () => { }; await createMemberResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(MEMBER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(MEMBER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -173,4 +176,31 @@ describe("resolvers -> Mutation -> createAdmin", () => { expect(updatedOrganizationCheck).toBe(true); }); + it("throws error if the user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + try { + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + + const args: MutationCreateMemberArgs = { + input: { + organizationId: testOrganization?.id, + userId: testUser?.id, + }, + }; + + const context = { + userId: testUser?.id, + }; + + await createMemberResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/createMessageChat.spec.ts b/tests/resolvers/Mutation/createMessageChat.spec.ts index e1033a6880..a3c77177d4 100644 --- a/tests/resolvers/Mutation/createMessageChat.spec.ts +++ b/tests/resolvers/Mutation/createMessageChat.spec.ts @@ -1,25 +1,29 @@ import "dotenv/config"; -import type { Document } from "mongoose"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import type { InterfaceUser, InterfaceMessageChat } from "../../../src/models"; -import { User } from "../../../src/models"; +import type { InterfaceMessageChat } from "../../../src/models"; +import { AppUserProfile, User } from "../../../src/models"; import type { MutationCreateMessageChatArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; import { nanoid } from "nanoid"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; +import { + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import type { TestUserType } from "../../helpers/userAndOrg"; -let testUsers: (InterfaceUser & Document)[]; +let testUsers: TestUserType[]; +// let testAppUserProfile: TestAppUserProfileType[]; let MONGOOSE_INSTANCE: typeof mongoose; beforeAll(async () => { @@ -31,16 +35,18 @@ beforeAll(async () => { password: "password", firstName: "firstName", lastName: "lastName", - appLanguageCode: "en", }, { email: `email${nanoid().toLowerCase()}@gmail.com`, password: "password", firstName: "firstName", lastName: "lastName", - appLanguageCode: "en", }, ]); + const appUserProfiles = testUsers.map((user) => ({ + userId: user?._id, + })); + await AppUserProfile.insertMany(appUserProfiles); }); afterAll(async () => { @@ -68,26 +74,128 @@ describe("resolvers -> Mutation -> createMessageChat", () => { }; const context = { - userId: testUsers[0].id, + userId: testUsers[0]?.id, + }; + + const { createMessageChat: createMessageChatResolver } = await import( + "../../../src/resolvers/Mutation/createMessageChat" + ); + await createMessageChatResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` + ); + } + }); + it(`throws user not found error if no user exist with id==context.userId`, async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + + try { + const args: MutationCreateMessageChatArgs = { + data: { + message: "", + receiver: testUsers[1]?.id, + }, + }; + + const context = { + userId: Types.ObjectId().toString(), }; const { createMessageChat: createMessageChatResolver } = await import( "../../../src/resolvers/Mutation/createMessageChat" ); await createMessageChatResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { + // console.log(error); expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } }); + it("throws error if receiver user does not have appProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + + try { + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const args: MutationCreateMessageChatArgs = { + data: { + message: "", + receiver: newUser.id, + }, + }; + + const context = { + userId: testUsers[0]?.id, + }; + + const { createMessageChat: createMessageChatResolver } = await import( + "../../../src/resolvers/Mutation/createMessageChat" + ); + await createMessageChatResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); + it("throws error if sender user does not have appProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + + try { + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const args: MutationCreateMessageChatArgs = { + data: { + message: "", + receiver: testUsers[1]?.id, + }, + }; + + const context = { + userId: newUser.id, + }; + + const { createMessageChat: createMessageChatResolver } = await import( + "../../../src/resolvers/Mutation/createMessageChat" + ); + await createMessageChatResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); it(`creates the organization and returns it`, async () => { const args: MutationCreateMessageChatArgs = { data: { message: "message", - receiver: testUsers[1].id, + receiver: testUsers[1]?.id, }, }; @@ -106,7 +214,7 @@ describe("resolvers -> Mutation -> createMessageChat", () => { }; const context = { - userId: testUsers[0].id, + userId: testUsers[0]?.id, pubsub, }; @@ -121,8 +229,8 @@ describe("resolvers -> Mutation -> createMessageChat", () => { expect(createMessageChatPayload).toEqual( expect.objectContaining({ - sender: testUsers[0]._id, - receiver: testUsers[1]._id, + sender: testUsers[0]?._id, + receiver: testUsers[1]?._id, message: "message", languageBarrier: false, }) diff --git a/tests/resolvers/Mutation/createOrganization.spec.ts b/tests/resolvers/Mutation/createOrganization.spec.ts index 03e107065d..e61a76298d 100644 --- a/tests/resolvers/Mutation/createOrganization.spec.ts +++ b/tests/resolvers/Mutation/createOrganization.spec.ts @@ -1,6 +1,6 @@ import "dotenv/config"; import type mongoose from "mongoose"; -import { ActionItemCategory, User } from "../../../src/models"; +import { ActionItemCategory, AppUserProfile, User } from "../../../src/models"; import type { MutationCreateOrganizationArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; @@ -15,7 +15,9 @@ import { } from "vitest"; import { LENGTH_VALIDATION_ERROR, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, } from "../../../src/constants"; import { createOrganization as createOrganizationResolver } from "../../../src/resolvers/Mutation/createOrganization"; import * as uploadEncodedImage from "../../../src/utilities/encodedImageStorage/uploadEncodedImage"; @@ -79,11 +81,13 @@ describe("resolvers -> Mutation -> createOrganization", () => { "../../../src/resolvers/Mutation/createOrganization" ); await createOrganization?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith( USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE ); - expect(error.message).toEqual(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE + ); } }); @@ -106,7 +110,16 @@ describe("resolvers -> Mutation -> createOrganization", () => { { $set: { adminApproved: true, - userType: "SUPERADMIN", + }, + } + ); + await AppUserProfile.updateOne( + { + userId: testUser?._id, + }, + { + $set: { + isSuperAdmin: true, }, } ); @@ -168,16 +181,23 @@ describe("resolvers -> Mutation -> createOrganization", () => { const updatedTestUser = await User.findOne({ _id: testUser?._id, }) - .select(["joinedOrganizations", "createdOrganizations", "adminFor"]) + .select(["joinedOrganizations"]) + .lean(); + const updatedTestAppUser = await AppUserProfile.findOne({ + userId: testUser?._id, + }) + .select(["createdOrganizations", "adminFor"]) .lean(); - expect(updatedTestUser).toEqual( + expect(updatedTestAppUser).toEqual( expect.objectContaining({ - joinedOrganizations: [createOrganizationPayload?._id], createdOrganizations: [createOrganizationPayload?._id], adminFor: [createOrganizationPayload?._id], }) ); + expect(updatedTestUser?.joinedOrganizations).toContainEqual( + createOrganizationPayload?._id + ); const defaultCategory = await ActionItemCategory.findOne({ organizationId: createOrganizationPayload?._id, @@ -193,9 +213,9 @@ describe("resolvers -> Mutation -> createOrganization", () => { }); it(`creates the organization without image and returns it`, async () => { vi.spyOn(uploadImage, "uploadImage").mockImplementation( - async (newImagePath: any, imageAlreadyInDbPath: any) => ({ + async (newImagePath, imageAlreadyInDbPath) => ({ newImagePath, - imageAlreadyInDbPath, + imageAlreadyInDbPath: imageAlreadyInDbPath || "", }) ); const args: MutationCreateOrganizationArgs = { @@ -283,8 +303,8 @@ describe("resolvers -> Mutation -> createOrganization", () => { }; await createOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 256 characters in name` ); } @@ -321,8 +341,8 @@ describe("resolvers -> Mutation -> createOrganization", () => { }; await createOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 500 characters in description` ); } @@ -386,17 +406,17 @@ describe("resolvers -> Mutation -> createOrganization", () => { // Testing for Invalid address try { await createOrganizationResolver({}, invalidArgs, context); - } catch (error: any) { + } catch (error: unknown) { // Validate that the error message matches the expected Address Validation Error message - expect(error.message).toEqual("Not a Valid Address"); + expect((error as Error).message).toEqual("Not a Valid Address"); } //Testing for Valid address try { await createOrganizationResolver({}, validArgs, context); - } catch (error: any) { + } catch (error: unknown) { // Validate that the error message matches the expected Address Validation Error message - expect(error.message).toEqual("Something went wrong."); + expect((error as Error).message).toEqual("Something went wrong."); } } else { console.error( @@ -430,9 +450,9 @@ describe("resolvers -> Mutation -> createOrganization", () => { if (createOrganizationResolver) { try { await createOrganizationResolver({}, validArgs, context); - } catch (error: any) { + } catch (error: unknown) { // Validate that the error message matches the expected Address Validation Error message - expect(error.message).toEqual("Not a Valid Address"); + expect((error as Error).message).toEqual("Not a Valid Address"); } } else { console.error( @@ -440,4 +460,77 @@ describe("resolvers -> Mutation -> createOrganization", () => { ); } }); + it("throws error if no user is found", async () => { + try { + const args: MutationCreateOrganizationArgs = { + data: { + description: "description", + name: "name", + userRegistrationRequired: true, + visibleInSearch: true, + apiUrl: "apiUrl", + address: { + city: "CityName", + countryCode: "US", + dependentLocality: "Dependent Locality", + line1: "123 Main Street", + line2: "Apartment 456", + postalCode: "12345", + sortingCode: "ABC-123", + state: "State/Province", + }, + }, + file: null, + }; + + const context = { + userId: "randomUserId", + }; + + await createOrganizationResolver?.({}, args, context); + } catch (error: unknown) { + console.log(error); + // expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } + }); + it("throws error if user does not have appProfile", async () => { + try { + const temp = await createTestUserFunc(); + await AppUserProfile.deleteOne({ + userId: temp?._id, + }); + const args: MutationCreateOrganizationArgs = { + data: { + description: "description", + name: "name", + userRegistrationRequired: true, + visibleInSearch: true, + apiUrl: "apiUrl", + address: { + city: "CityName", + countryCode: "US", + dependentLocality: "Dependent Locality", + line1: "123 Main Street", + line2: "Apartment 456", + postalCode: "12345", + sortingCode: "ABC-123", + state: "State/Province", + }, + }, + file: null, + }; + const context = { + userId: temp?._id, + }; + + await createOrganizationResolver?.({}, args, context); + } catch (error: unknown) { + // console.log((error as Error).message); + + expect((error as Error).message).toEqual( + `${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/createPost.spec.ts b/tests/resolvers/Mutation/createPost.spec.ts index c3a69250b4..9f779c1646 100644 --- a/tests/resolvers/Mutation/createPost.spec.ts +++ b/tests/resolvers/Mutation/createPost.spec.ts @@ -5,32 +5,34 @@ import type { MutationCreatePostArgs } from "../../../src/types/generatedGraphQL import { connect, disconnect } from "../../helpers/db"; import { - USER_NOT_FOUND_ERROR, - ORGANIZATION_NOT_FOUND_ERROR, - LENGTH_VALIDATION_ERROR, - USER_NOT_AUTHORIZED_TO_PIN, - BASE_URL, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; +import { + BASE_URL, + LENGTH_VALIDATION_ERROR, + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_AUTHORIZED_TO_PIN, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile, Organization, Post } from "../../../src/models"; +import { createPost as createPostResolverImage } from "../../../src/resolvers/Mutation/createPost"; +import * as uploadEncodedImage from "../../../src/utilities/encodedImageStorage/uploadEncodedImage"; +import * as uploadEncodedVideo from "../../../src/utilities/encodedVideoStorage/uploadEncodedVideo"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; import { - createTestUserAndOrganization, createTestUser, + createTestUserAndOrganization, } from "../../helpers/userAndOrg"; -import { Organization } from "../../../src/models"; -import * as uploadEncodedImage from "../../../src/utilities/encodedImageStorage/uploadEncodedImage"; -import { createPost as createPostResolverImage } from "../../../src/resolvers/Mutation/createPost"; let testUser: TestUserType; let randomUser: TestUserType; @@ -84,9 +86,9 @@ describe("resolvers -> Mutation -> createPost", () => { ); await createPostResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -117,9 +119,9 @@ describe("resolvers -> Mutation -> createPost", () => { ); await createPostResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${ORGANIZATION_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -146,14 +148,15 @@ describe("resolvers -> Mutation -> createPost", () => { userId: randomUser?.id, }; + expect(args.data.pinned).toBe(true); const { createPost: createPostResolver } = await import( "../../../src/resolvers/Mutation/createPost" ); await createPostResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_TO_PIN.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_TO_PIN.MESSAGE}` ); } @@ -179,16 +182,18 @@ describe("resolvers -> Mutation -> createPost", () => { userId: testUser?.id, }; + expect(args.data.pinned).toBe(true); + const { createPost: createPostResolver } = await import( "../../../src/resolvers/Mutation/createPost" ); const createdPost = await createPostResolver?.({}, args, context); - expect(createdPost).toEqual( expect.objectContaining({ text: "New Post Text", videoUrl: null, // Update the expected value to match the received value title: "New Post Title", + pinned: true, }) ); @@ -203,7 +208,7 @@ describe("resolvers -> Mutation -> createPost", () => { ).toBeTruthy(); }); - it(`creates the post and returns it when image is not provided`, async () => { + it(`creates the post and returns it when image or video is not provided`, async () => { const args: MutationCreatePostArgs = { data: { organizationId: testOrganization?.id, @@ -218,11 +223,14 @@ describe("resolvers -> Mutation -> createPost", () => { userId: testUser?.id, }; + expect(args.data.pinned).toBe(true); + const { createPost: createPostResolver } = await import( "../../../src/resolvers/Mutation/createPost" ); const createPostPayload = await createPostResolver?.({}, args, context); + expect(createPostPayload?.pinned).toBe(true); expect(createPostPayload).toEqual( expect.objectContaining({ @@ -235,6 +243,82 @@ describe("resolvers -> Mutation -> createPost", () => { ); }); + it(`creates the post and and returns it when an image is provided`, async () => { + const args: MutationCreatePostArgs = { + data: { + organizationId: testOrganization?.id, + text: "text", + title: "title", + pinned: true, + }, + file: "", // Provide a supported file type + }; + + vi.spyOn(uploadEncodedImage, "uploadEncodedImage").mockImplementation( + async (encodedImageURL: string) => encodedImageURL + ); + + const context = { + userId: testUser?.id, + apiRootUrl: BASE_URL, + }; + + expect(args.data.pinned).toBe(true); + + const createPostPayload = await createPostResolverImage?.( + {}, + args, + context + ); + expect(createPostPayload?.pinned).toBe(true); + + const testCreatePostPayload = await Post.findOne({ + _id: createPostPayload?._id, + imageUrl: { $ne: null }, + }).lean(); + + //Ensures that the post is created and imageUrl is not null + expect(testCreatePostPayload).not.toBeNull(); + }); + + it(`creates the post and and returns it when a video is provided`, async () => { + const args: MutationCreatePostArgs = { + data: { + organizationId: testOrganization?.id, + text: "text", + title: "title", + pinned: true, + }, + file: "data:video/mp4;base64,VIDEO_BASE64_DATA_HERE", // Provide a supported file type + }; + + vi.spyOn(uploadEncodedVideo, "uploadEncodedVideo").mockImplementation( + async (uploadEncodedVideo: string) => uploadEncodedVideo + ); + + const context = { + userId: testUser?.id, + apiRootUrl: BASE_URL, + }; + + expect(args.data.pinned).toBe(true); + + const createPostPayload = await createPostResolverImage?.( + {}, + args, + context + ); + expect(createPostPayload?.pinned).toBe(true); + + const testCreatePostPayload = await Post.findOne({ + _id: createPostPayload?._id, + videoUrl: { $ne: null }, + }).lean(); + + //Ensures that the post is created and videoUrl is not null + expect(testCreatePostPayload).not.toBeNull(); + }); + it(`creates the post and throws an error for unsupported file type`, async () => { const args: MutationCreatePostArgs = { data: { @@ -242,6 +326,7 @@ describe("resolvers -> Mutation -> createPost", () => { text: "text", videoUrl: "videoUrl", title: "title", + pinned: true, }, file: "unsupportedFile.txt", // Provide an unsupported file type }; @@ -251,6 +336,8 @@ describe("resolvers -> Mutation -> createPost", () => { apiRootUrl: BASE_URL, }; + expect(args.data.pinned).toBe(true); + // Mock the uploadEncodedImage function to throw an error for unsupported file types vi.spyOn(uploadEncodedImage, "uploadEncodedImage").mockImplementation( () => { @@ -285,14 +372,14 @@ describe("resolvers -> Mutation -> createPost", () => { const context = { userId: testUser?.id, }; - + expect(args.data.pinned).toBe(true); const { createPost: createPostResolver } = await import( "../../../src/resolvers/Mutation/createPost" ); await createPostResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 256 characters in title` ); } @@ -318,13 +405,15 @@ describe("resolvers -> Mutation -> createPost", () => { userId: testUser?.id, }; + expect(args.data.pinned).toBe(true); + const { createPost: createPostResolver } = await import( "../../../src/resolvers/Mutation/createPost" ); await createPostResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 500 characters in information` ); } @@ -349,12 +438,13 @@ describe("resolvers -> Mutation -> createPost", () => { userId: testUser?.id, }; + expect(args.data.pinned).toBe(false); const { createPost: createPostResolver } = await import( "../../../src/resolvers/Mutation/createPost" ); await createPostResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Post needs to be pinned inorder to add a title` ); } @@ -378,13 +468,50 @@ describe("resolvers -> Mutation -> createPost", () => { const context = { userId: testUser?.id, }; + expect(args.data.pinned).toBe(true); const { createPost: createPostResolver } = await import( "../../../src/resolvers/Mutation/createPost" ); await createPostResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(`Please provide a title to pin post`); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `Please provide a title to pin post` + ); + } + }); + it("throws error if the user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + await Organization.deleteOne({ + _id: testOrganization?._id, + }); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const args: MutationCreatePostArgs = { + data: { + organizationId: testOrganization?._id, + title: "Test title", + text: "Test text", + pinned: true, + }, + }; + const context = { + userId: testUser?._id, + }; + try { + const { createPost: createPostResolver } = await import( + "../../../src/resolvers/Mutation/createPost" + ); + await createPostResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); } }); }); diff --git a/tests/resolvers/Mutation/createSampleOrganization.spec.ts b/tests/resolvers/Mutation/createSampleOrganization.spec.ts index c378e5465c..8547e00328 100644 --- a/tests/resolvers/Mutation/createSampleOrganization.spec.ts +++ b/tests/resolvers/Mutation/createSampleOrganization.spec.ts @@ -1,25 +1,25 @@ +import { faker } from "@faker-js/faker"; +import type mongoose from "mongoose"; import { afterAll, beforeAll, + beforeEach, describe, expect, it, vi, - beforeEach, } from "vitest"; -import { generateUserData } from "../../../src/utilities/createSampleOrganizationUtil"; -import { faker } from "@faker-js/faker"; +import { SampleData, User } from "../../../src/models"; import { createSampleOrganization } from "../../../src/resolvers/Mutation/createSampleOrganization"; -import type mongoose from "mongoose"; -import { SampleData } from "../../../src/models"; +import { generateUserData } from "../../../src/utilities/createSampleOrganizationUtil"; +import { Types } from "mongoose"; +import { nanoid } from "nanoid"; import { - SAMPLE_ORGANIZATION_ALREADY_EXISTS, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; import { connect, disconnect } from "../../helpers/db"; -import { Types } from "mongoose"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -43,20 +43,21 @@ describe("createSampleOrganization resolver", async () => { ); const ORGANIZATION_ID = faker.database.mongodbObjectId(); - const admin = await generateUserData(ORGANIZATION_ID, "ADMIN"); - admin.save(); + const userData = await generateUserData(ORGANIZATION_ID, "ADMIN"); + const admin = userData.user; const args = {}; const adminContext = { userId: admin._id }; const parent = {}; - const adminResult = await createSampleOrganization!( - parent, - args, - adminContext - ); - expect(adminResult).toBe(true); - await SampleData.deleteMany({}); + try { + if (createSampleOrganization) { + await createSampleOrganization(parent, args, adminContext); + } + } catch (error: unknown) { + expect((error as Error).message).toBe(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + await SampleData.deleteMany({}); + } }); it("should NOT throw error when user is SUPERADMIN", async () => { @@ -66,19 +67,21 @@ describe("createSampleOrganization resolver", async () => { ); const ORGANIZATION_ID = faker.database.mongodbObjectId(); - const admin = await generateUserData(ORGANIZATION_ID, "SUPERADMIN"); - admin.save(); + const userData = await generateUserData(ORGANIZATION_ID, "SUPERADMIN"); + const admin = userData.user; const args = {}; const adminContext = { userId: admin._id }; const parent = {}; - - const adminResult = await createSampleOrganization!( - parent, - args, - adminContext - ); - expect(adminResult).toBe(true); + if (createSampleOrganization) { + const adminResult = await createSampleOrganization( + parent, + args, + adminContext + ); + + expect(adminResult).toBe(true); + } await SampleData.deleteMany({}); }); @@ -89,67 +92,66 @@ describe("createSampleOrganization resolver", async () => { ); const ORGANIZATION_ID = faker.database.mongodbObjectId(); - const admin = await generateUserData(ORGANIZATION_ID, "USER"); - admin.save(); + const userData = await generateUserData(ORGANIZATION_ID, "USER"); + const admin = userData.user; const args = {}; const adminContext = { userId: admin._id }; const parent = {}; try { - await createSampleOrganization!(parent, args, adminContext); - } catch (error: any) { - expect(error.message).toBe(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + if (createSampleOrganization) { + await createSampleOrganization(parent, args, adminContext); + } + } catch (error: unknown) { + expect((error as Error).message).toBe(USER_NOT_AUTHORIZED_ERROR.MESSAGE); } }); - it("should throw error when the sample organization already exist", async () => { + it("should throw error when the current user doesn't exist", async () => { const { requestContext } = await import("../../../src/libraries"); vi.spyOn(requestContext, "translate").mockImplementation( (message) => message ); - const ORGANIZATION_ID = faker.database.mongodbObjectId(); - const admin = await generateUserData(ORGANIZATION_ID, "ADMIN"); - admin.save(); - const args = {}; - const adminContext = { userId: admin._id }; + const adminContext = { userId: Types.ObjectId().toString() }; const parent = {}; - const adminResult = await createSampleOrganization!( - parent, - args, - adminContext - ); - - expect(adminResult).toBe(true); - try { - await createSampleOrganization!(parent, args, adminContext); - } catch (error: any) { - expect(error.message).toBe(SAMPLE_ORGANIZATION_ALREADY_EXISTS.MESSAGE); + if (createSampleOrganization) + await createSampleOrganization(parent, args, adminContext); + } catch (error: unknown) { + expect((error as Error).message).toBe(USER_NOT_FOUND_ERROR.MESSAGE); } await SampleData.deleteMany({}); }); - - it("should throw error when the current user doesn't exist", async () => { + it("throws error if user does not have AppUserProfile", async () => { const { requestContext } = await import("../../../src/libraries"); - vi.spyOn(requestContext, "translate").mockImplementation( - (message) => message - ); - + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); const args = {}; - const adminContext = { userId: Types.ObjectId().toString() }; - const parent = {}; + const context = { + userId: newUser.id, + }; try { - await createSampleOrganization!(parent, args, adminContext); - } catch (error: any) { - expect(error.message).toBe(USER_NOT_FOUND_ERROR.MESSAGE); + if (createSampleOrganization) + await createSampleOrganization({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); } - - await SampleData.deleteMany({}); }); }); diff --git a/tests/resolvers/Mutation/createUserFamily.spec.ts b/tests/resolvers/Mutation/createUserFamily.spec.ts index dde3a66dde..b095606793 100644 --- a/tests/resolvers/Mutation/createUserFamily.spec.ts +++ b/tests/resolvers/Mutation/createUserFamily.spec.ts @@ -4,16 +4,18 @@ import { Types } from "mongoose"; import type { MutationCreateUserFamilyArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { LENGTH_VALIDATION_ERROR, USER_FAMILY_MIN_MEMBERS_ERROR_CODE, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_AUTHORIZED_SUPERADMIN, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; +import { AppUserProfile } from "../../../src/models"; +import { createTestUserFunc as createTestUser } from "../../helpers/user"; import type { TestUserType } from "../../helpers/userAndUserFamily"; import { createTestUserFunc } from "../../helpers/userAndUserFamily"; -import { createTestUserFunc as createTestUser } from "../../helpers/user"; let testUser: TestUserType; let testUser2: TestUserType; @@ -186,4 +188,28 @@ describe("resolvers -> Mutation -> createUserFamily", () => { }) ); }); + it("throws an error if the user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const args: MutationCreateUserFamilyArgs = { + data: { + title: "title", + userIds: [testUser2?._id, testUser?._id], + }, + }; + const context = { + userId: testUser?._id, + }; + const { createUserFamily: createUserFamilyResolver } = await import( + "../../../src/resolvers/Mutation/createUserFamily" + ); + try { + await createUserFamilyResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/createUserTag.spec.ts b/tests/resolvers/Mutation/createUserTag.spec.ts index a72ea942e7..595ca8ec58 100644 --- a/tests/resolvers/Mutation/createUserTag.spec.ts +++ b/tests/resolvers/Mutation/createUserTag.spec.ts @@ -5,30 +5,31 @@ import type { MutationCreateUserTagArgs } from "../../../src/types/generatedGrap import { connect, disconnect } from "../../helpers/db"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_TO_CREATE_TAG, - INCORRECT_TAG_INPUT, - ORGANIZATION_NOT_FOUND_ERROR, - TAG_NOT_FOUND, - TAG_ALREADY_EXISTS, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; +import { + INCORRECT_TAG_INPUT, + ORGANIZATION_NOT_FOUND_ERROR, + TAG_ALREADY_EXISTS, + TAG_NOT_FOUND, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_AUTHORIZED_TO_CREATE_TAG, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile, OrganizationTagUser } from "../../../src/models"; +import type { TestUserTagType } from "../../helpers/tags"; +import { createRootTagWithOrg } from "../../helpers/tags"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; import { createTestUser } from "../../helpers/userAndOrg"; -import { OrganizationTagUser } from "../../../src/models"; -import type { TestUserTagType } from "../../helpers/tags"; -import { createRootTagWithOrg } from "../../helpers/tags"; let testUser: TestUserType; let randomUser: TestUserType; @@ -79,9 +80,9 @@ describe("resolvers -> Mutation -> createUserTag", () => { ); await createUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -110,9 +111,9 @@ describe("resolvers -> Mutation -> createUserTag", () => { ); await createUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${ORGANIZATION_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -142,9 +143,11 @@ describe("resolvers -> Mutation -> createUserTag", () => { ); await createUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(TAG_NOT_FOUND.MESSAGE); - expect(error.message).toEqual(`Translated ${TAG_NOT_FOUND.MESSAGE}`); + expect((error as Error).message).toEqual( + `Translated ${TAG_NOT_FOUND.MESSAGE}` + ); } }); @@ -172,9 +175,9 @@ describe("resolvers -> Mutation -> createUserTag", () => { ); await createUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(INCORRECT_TAG_INPUT.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${INCORRECT_TAG_INPUT.MESSAGE}` ); } @@ -204,9 +207,9 @@ describe("resolvers -> Mutation -> createUserTag", () => { ); await createUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_TO_CREATE_TAG.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_TO_CREATE_TAG.MESSAGE}` ); } @@ -236,9 +239,11 @@ describe("resolvers -> Mutation -> createUserTag", () => { ); await createUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(TAG_ALREADY_EXISTS.MESSAGE); - expect(error.message).toEqual(`Translated ${TAG_ALREADY_EXISTS.MESSAGE}`); + expect((error as Error).message).toEqual( + `Translated ${TAG_ALREADY_EXISTS.MESSAGE}` + ); } }); @@ -280,4 +285,34 @@ describe("resolvers -> Mutation -> createUserTag", () => { expect(createdTagExists).toBeTruthy(); }); + it("throws an error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + try { + const args: MutationCreateUserTagArgs = { + input: { + organizationId: testOrganization?._id, + name: "TestUserTag", + }, + }; + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const context = { + userId: newUser?.id, + }; + const { createUserTag: createUserTagResolver } = await import( + "../../../src/resolvers/Mutation/createUserTag" + ); + await createUserTagResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/leaveOrganization.spec.ts b/tests/resolvers/Mutation/leaveOrganization.spec.ts index bf2ba6e0ba..8f50629f63 100644 --- a/tests/resolvers/Mutation/leaveOrganization.spec.ts +++ b/tests/resolvers/Mutation/leaveOrganization.spec.ts @@ -1,31 +1,32 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Organization } from "../../../src/models"; +import { AppUserProfile, Organization, User } from "../../../src/models"; import type { MutationLeaveOrganizationArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { leaveOrganization as leaveOrganizationResolver } from "../../../src/resolvers/Mutation/leaveOrganization"; -import { - MEMBER_NOT_FOUND_ERROR, - ORGANIZATION_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; +import { + MEMBER_NOT_FOUND_ERROR, + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { leaveOrganization as leaveOrganizationResolver } from "../../../src/resolvers/Mutation/leaveOrganization"; +import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; import { createTestUserAndOrganization } from "../../helpers/userAndOrg"; -import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -66,9 +67,11 @@ describe("resolvers -> Mutation -> leaveOrganization", () => { ); await leaveOrganizationResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -91,9 +94,9 @@ describe("resolvers -> Mutation -> leaveOrganization", () => { ); await leaveOrganizationResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -136,9 +139,9 @@ describe("resolvers -> Mutation -> leaveOrganization", () => { ); await leaveOrganizationResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(MEMBER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(MEMBER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(MEMBER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -192,4 +195,30 @@ describe("resolvers -> Mutation -> leaveOrganization", () => { expect(updatedTestOrganization?.admins).toEqual([]); expect(updatedTestOrganization?.members).toEqual([]); }); + it("throws error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const args: MutationLeaveOrganizationArgs = { + organizationId: testOrganization?.id, + }; + const context = { + userId: testUser?._id, + }; + try { + const { leaveOrganization: leaveOrganizationResolver } = await import( + "../../../src/resolvers/Mutation/leaveOrganization" + ); + await leaveOrganizationResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/login.spec.ts b/tests/resolvers/Mutation/login.spec.ts index a7917d651c..f2deea0b70 100644 --- a/tests/resolvers/Mutation/login.spec.ts +++ b/tests/resolvers/Mutation/login.spec.ts @@ -1,26 +1,31 @@ +import bcrypt from "bcryptjs"; import "dotenv/config"; -import { User, Organization, MembershipRequest } from "../../../src/models"; -import type { MutationLoginArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; -import { login as loginResolver } from "../../../src/resolvers/Mutation/login"; -import { - INVALID_CREDENTIALS_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; -import bcrypt from "bcryptjs"; import { nanoid } from "nanoid"; import { - beforeAll, afterAll, afterEach, + beforeAll, describe, - it, expect, + it, vi, } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; +import { + INVALID_CREDENTIALS_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { + AppUserProfile, + MembershipRequest, + Organization, + User, +} from "../../../src/models"; +import { login as loginResolver } from "../../../src/resolvers/Mutation/login"; +import type { MutationLoginArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import { createTestEventWithRegistrants } from "../../helpers/eventsWithRegistrants"; +import type { TestUserType } from "../../helpers/userAndOrg"; let testUser: TestUserType; let MONGOOSE_INSTANCE: typeof mongoose; @@ -108,6 +113,47 @@ describe("resolvers -> Mutation -> login", () => { } } }); + it("creates a appUserProfile of the user if does not exist", async () => { + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: "password", + firstName: "firstName", + lastName: "lastName", + }); + // console.log(newUser); + const hashedTestPassword = await bcrypt.hash("password", 12); + await User.updateOne( + { + _id: newUser?._id, + }, + { + $set: { + password: hashedTestPassword, + }, + } + ); + const args: MutationLoginArgs = { + data: { + email: newUser?.email, + password: "password", + }, + }; + + const loginPayload = await loginResolver?.({}, args, {}); + + const userAppProfile = await AppUserProfile.findOne({ + userId: newUser?._id, + }); + + expect(userAppProfile).toBeDefined(); + expect(loginPayload).toEqual( + expect.objectContaining({ + user: expect.objectContaining({ + appUserProfileId: userAppProfile?._id, + }), + }) + ); + }); it(`throws ValidationError if args.data.password !== password for user with email === args.data.email`, async () => { @@ -161,18 +207,18 @@ email === args.data.email`, async () => { const loginPayload = await loginResolver?.({}, args, {}); expect(await loginPayload?.user).toBeDefined(); - expect((await loginPayload?.user)?.userType).toEqual("SUPERADMIN"); + // expect((await loginPayload?.user)?.userType).toEqual("SUPERADMIN"); }); it("should update the user's token and increment the tokenVersion", async () => { const newToken = "new-token"; - const mockUser = await User.findOne({ - _id: testUser?._id, + const mockUserAppProfile = await AppUserProfile.findOne({ + userId: testUser?._id, }).lean(); - const updatedUser = await User.findOneAndUpdate( - { _id: testUser?._id }, + const updatedUser = await AppUserProfile.findOneAndUpdate( + { userId: testUser?._id }, { token: newToken, $inc: { tokenVersion: 1 } }, { new: true } ); @@ -180,13 +226,15 @@ email === args.data.email`, async () => { expect(updatedUser).toBeDefined(); expect(updatedUser?.token).toBe(newToken); - if (mockUser?.tokenVersion !== undefined) { - expect(updatedUser?.tokenVersion).toBe(mockUser?.tokenVersion + 1); + if (mockUserAppProfile?.tokenVersion !== undefined) { + expect(updatedUser?.tokenVersion).toBe( + mockUserAppProfile?.tokenVersion + 1 + ); } }); - it(`returns the user object with populated fields joinedOrganizations, createdOrganizations, - createdEvents, registeredEvents, eventAdmin, adminFor, membershipRequests, + it(`returns the user object with populated fields joinedOrganizations, + registeredEvents, membershipRequests, organizationsBlockedBy`, async () => { const args: MutationLoginArgs = { data: { @@ -202,11 +250,7 @@ email === args.data.email`, async () => { }) .select(["-password"]) .populate("joinedOrganizations") - .populate("createdOrganizations") - .populate("createdEvents") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .populate("membershipRequests") .populate("organizationsBlockedBy") .lean(); diff --git a/tests/resolvers/Mutation/logout.spec.ts b/tests/resolvers/Mutation/logout.spec.ts index 6391ecf6bb..5da8bab3da 100644 --- a/tests/resolvers/Mutation/logout.spec.ts +++ b/tests/resolvers/Mutation/logout.spec.ts @@ -1,18 +1,18 @@ import "dotenv/config"; import type mongoose from "mongoose"; -import { User } from "../../../src/models"; +import { AppUserProfile } from "../../../src/models"; import { connect, disconnect } from "../../helpers/db"; -import { logout as logoutResolver } from "../../../src/resolvers/Mutation/logout"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; +import { logout as logoutResolver } from "../../../src/resolvers/Mutation/logout"; import type { TestUserType } from "../../helpers/userAndOrg"; import { createTestUserAndOrganization } from "../../helpers/userAndOrg"; @@ -44,12 +44,11 @@ describe("resolvers -> Mutation -> logout", () => { expect(logoutPayload).toEqual(true); - const updatedTestUser = await User.findOne({ - _id: testUser?._id, + const updateTestAppUserProfile = await AppUserProfile.findOne({ + userId: testUser?._id, }) .select(["token"]) .lean(); - - expect(updatedTestUser?.token).toEqual(null); + expect(updateTestAppUserProfile?.token).toEqual(null); }); }); diff --git a/tests/resolvers/Mutation/refreshToken.spec.ts b/tests/resolvers/Mutation/refreshToken.spec.ts index c6030aa532..1190e94bd3 100644 --- a/tests/resolvers/Mutation/refreshToken.spec.ts +++ b/tests/resolvers/Mutation/refreshToken.spec.ts @@ -1,35 +1,45 @@ import "dotenv/config"; import type mongoose from "mongoose"; -import { Types } from "mongoose"; -import type { InterfaceUser } from "../../../src/models"; -import { User } from "../../../src/models"; +import type { + InterfaceAppUserProfile, + InterfaceUser, +} from "../../../src/models"; +import { AppUserProfile, User } from "../../../src/models"; import type { MutationRefreshTokenArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; import { - INVALID_REFRESH_TOKEN_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; -import { createRefreshToken } from "../../../src/utilities"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; -import type { TestUserType } from "../../helpers/user"; +import { + INVALID_REFRESH_TOKEN_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { createRefreshToken } from "../../../src/utilities"; import { createTestUserFunc } from "../../helpers/user"; +import type { + TestAppUserProfileType, + TestUserType, +} from "../../helpers/userAndOrg"; let testUser: TestUserType; +let testUserAppProfile: TestAppUserProfileType; let refreshToken: string; let MONGOOSE_INSTANCE: typeof mongoose; beforeAll(async () => { MONGOOSE_INSTANCE = await connect(); testUser = await createTestUserFunc(); + testUserAppProfile = await AppUserProfile.findOne({ + userId: testUser?._id, + }).lean(); }); afterAll(async () => { @@ -74,10 +84,12 @@ describe("resolvers -> Mutation -> refreshToken", () => { .spyOn(requestContext, "translate") .mockImplementation((message) => `Translated ${message}`); try { - refreshToken = await createRefreshToken({ - ...testUser!.toObject(), - _id: Types.ObjectId(), - }); + refreshToken = await createRefreshToken( + testUser ? testUser.toObject() : ({} as InterfaceUser), + testUserAppProfile + ? testUserAppProfile + : ({} as InterfaceAppUserProfile) + ); const args: MutationRefreshTokenArgs = { refreshToken, @@ -103,13 +115,13 @@ describe("resolvers -> Mutation -> refreshToken", () => { it(`throws ValidationError if user.tokenVersion !== payload.tokenVersion for user with _id === payload.userId for args.refreshToken`, async () => { const { requestContext } = await import("../../../src/libraries"); - vi.spyOn(requestContext, "translate").mockImplementation( - (message) => message - ); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementation((message) => `Translated ${message}`); try { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $inc: { @@ -119,7 +131,10 @@ describe("resolvers -> Mutation -> refreshToken", () => { ); refreshToken = await createRefreshToken( - testUser ? testUser.toObject() : ({} as InterfaceUser) + testUser ? testUser.toObject() : ({} as InterfaceUser), + testUserAppProfile + ? testUserAppProfile + : ({} as InterfaceAppUserProfile) ); const args: MutationRefreshTokenArgs = { @@ -133,7 +148,10 @@ describe("resolvers -> Mutation -> refreshToken", () => { await refreshTokenResolver?.({}, args, {}); } catch (error: unknown) { if (error instanceof Error) { - expect(error.message).toEqual(INVALID_REFRESH_TOKEN_ERROR.MESSAGE); + expect(spy).toBeCalledWith(INVALID_REFRESH_TOKEN_ERROR.MESSAGE); + expect(error.message).toEqual( + `Translated ${INVALID_REFRESH_TOKEN_ERROR.MESSAGE}` + ); } } }); @@ -145,9 +163,9 @@ describe("resolvers -> Mutation -> refreshToken", () => { .spyOn(requestContext, "translate") .mockImplementation((message) => `Translated ${message}`); try { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $inc: { @@ -157,7 +175,10 @@ describe("resolvers -> Mutation -> refreshToken", () => { ); refreshToken = await createRefreshToken( - testUser ? testUser.toObject() : ({} as InterfaceUser) + testUser ? testUser.toObject() : ({} as InterfaceUser), + testUserAppProfile + ? testUserAppProfile + : ({} as InterfaceAppUserProfile) ); const args: MutationRefreshTokenArgs = { @@ -184,49 +205,52 @@ describe("resolvers -> Mutation -> refreshToken", () => { const newRefreshToken = "new-refresh-token"; // Save the original function - const originalFunction = User.findOneAndUpdate; + const originalFunction = AppUserProfile.findOneAndUpdate; // Replace User.findOneAndUpdate with a mock function /* eslint-disable */ - User.findOneAndUpdate = function () { + AppUserProfile.findOneAndUpdate = function () { return Promise.resolve({ - _id: testUser?._id, + userId: testUser?._id, token: newRefreshToken, - tokenVersion: testUser?.tokenVersion || 0 + 1, + tokenVersion: testUserAppProfile?.tokenVersion || 0 + 1, }); } as any; /* eslint-enable */ - const updatedUser = await User.findOneAndUpdate( - { _id: jwtPayload.userId }, + const updatedUserAppProfile = await AppUserProfile.findOneAndUpdate( + { userId: jwtPayload.userId }, { $set: { token: newRefreshToken }, $inc: { tokenVersion: 1 } }, { new: true } ); - expect(updatedUser).toBeDefined(); - expect(updatedUser?.token).toBe(newRefreshToken); - if (testUser?.tokenVersion) - expect(updatedUser?.tokenVersion).toBe(testUser?.tokenVersion + 1); + expect(updatedUserAppProfile).toBeDefined(); + expect(updatedUserAppProfile?.token).toBe(newRefreshToken); + if (testUserAppProfile?.tokenVersion) + expect(updatedUserAppProfile?.tokenVersion).toBe( + testUserAppProfile?.tokenVersion + 1 + ); // Restore the original function - User.findOneAndUpdate = originalFunction; + AppUserProfile.findOneAndUpdate = originalFunction; }); it(`generates new accessToken and refreshToken and returns them`, async () => { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $inc: { - tokenVersion: -2, + tokenVersion: -4, }, } ); refreshToken = await createRefreshToken( - testUser ? testUser.toObject() : ({} as InterfaceUser) + testUser ? testUser.toObject() : ({} as InterfaceUser), + testUserAppProfile ? testUserAppProfile : ({} as InterfaceAppUserProfile) ); - + // console.log(refreshToken) const args: MutationRefreshTokenArgs = { refreshToken, }; @@ -243,4 +267,76 @@ describe("resolvers -> Mutation -> refreshToken", () => { expect(typeof refreshTokenPayload?.refreshToken).toEqual("string"); expect(refreshTokenPayload?.refreshToken.length).toBeGreaterThan(1); }); + it("throws error if user does not exists", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementation((message) => `Translated ${message}`); + const newUser = await createTestUserFunc(); + const newUserAppProfile = await AppUserProfile.findOne({ + userId: newUser?._id, + }).lean(); + const newRefreshToken = await createRefreshToken( + newUser ? newUser.toObject() : ({} as InterfaceUser), + newUserAppProfile ? newUserAppProfile : ({} as InterfaceAppUserProfile) + ); + await User.deleteOne({ + _id: newUser?._id, + }); + await AppUserProfile.deleteOne({ + userId: newUser?._id, + }); + const args: MutationRefreshTokenArgs = { + refreshToken: newRefreshToken, + }; + const { refreshToken: refreshTokenResolver } = await import( + "../../../src/resolvers/Mutation/refreshToken" + ); + try { + await refreshTokenResolver?.({}, args, {}); + } catch (error: unknown) { + if (error instanceof Error) { + expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); + expect(error.message).toEqual( + `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` + ); + } + } + }); + it("throws error if user does not have appUserProfile ", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementation((message) => `Translated ${message}`); + const newUser = await createTestUserFunc(); + const newUserAppProfile = await AppUserProfile.findOne({ + userId: newUser?._id, + }).lean(); + const newRefreshToken = await createRefreshToken( + newUser ? newUser.toObject() : ({} as InterfaceUser), + newUserAppProfile ? newUserAppProfile : ({} as InterfaceAppUserProfile) + ); + // await User.deleteOne({ + // _id: newUser?._id, + // }); + await AppUserProfile.deleteOne({ + userId: newUser?._id, + }); + const args: MutationRefreshTokenArgs = { + refreshToken: newRefreshToken, + }; + const { refreshToken: refreshTokenResolver } = await import( + "../../../src/resolvers/Mutation/refreshToken" + ); + try { + await refreshTokenResolver?.({}, args, {}); + } catch (error: unknown) { + if (error instanceof Error) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect(error.message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + } + }); }); diff --git a/tests/resolvers/Mutation/registerForEvent.spec.ts b/tests/resolvers/Mutation/registerForEvent.spec.ts index 4137b1d3c1..ea43e17d42 100644 --- a/tests/resolvers/Mutation/registerForEvent.spec.ts +++ b/tests/resolvers/Mutation/registerForEvent.spec.ts @@ -1,27 +1,27 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Event, EventAttendee } from "../../../src/models"; +import { Event, EventAttendee, User } from "../../../src/models"; import type { MutationRegisterForEventArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { registerForEvent as registerForEventResolver } from "../../../src/resolvers/Mutation/registerForEvent"; -import { - EVENT_NOT_FOUND_ERROR, - REGISTRANT_ALREADY_EXIST_ERROR, -} from "../../../src/constants"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; +import { + EVENT_NOT_FOUND_ERROR, + REGISTRANT_ALREADY_EXIST_ERROR, +} from "../../../src/constants"; +import { registerForEvent as registerForEventResolver } from "../../../src/resolvers/Mutation/registerForEvent"; import type { TestEventType } from "../../helpers/events"; import { createTestEventWithRegistrants } from "../../helpers/eventsWithRegistrants"; +import type { TestUserType } from "../../helpers/userAndOrg"; let testUser: TestUserType; let MONGOOSE_INSTANCE: typeof mongoose; @@ -62,9 +62,9 @@ describe("resolvers -> Mutation -> registerForEvent", () => { ); await registerForEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -76,11 +76,11 @@ describe("resolvers -> Mutation -> registerForEvent", () => { try { const args: MutationRegisterForEventArgs = { - id: testEvent!._id, + id: testEvent?._id.toString() ?? "", }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { registerForEvent: registerForEventResolver } = await import( @@ -88,16 +88,18 @@ describe("resolvers -> Mutation -> registerForEvent", () => { ); await registerForEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(REGISTRANT_ALREADY_EXIST_ERROR.MESSAGE); - expect(error.message).toEqual(REGISTRANT_ALREADY_EXIST_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + REGISTRANT_ALREADY_EXIST_ERROR.MESSAGE + ); } }); it(`registers user with _id === context.userId as a registrant for event with _id === args.id`, async () => { await EventAttendee.deleteOne({ - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id, }); await User.updateOne( @@ -112,11 +114,11 @@ describe("resolvers -> Mutation -> registerForEvent", () => { ); const args: MutationRegisterForEventArgs = { - id: testEvent!._id, + id: testEvent?._id.toString() ?? "", }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const registerForEventPayload = await registerForEventResolver?.( diff --git a/tests/resolvers/Mutation/rejectAdmin.spec.ts b/tests/resolvers/Mutation/rejectAdmin.spec.ts index 9918eb86b6..016772d2df 100644 --- a/tests/resolvers/Mutation/rejectAdmin.spec.ts +++ b/tests/resolvers/Mutation/rejectAdmin.spec.ts @@ -1,24 +1,24 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User } from "../../../src/models"; +import { AppUserProfile } from "../../../src/models"; import type { MutationRejectAdminArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { rejectAdmin as rejectAdminResolver } from "../../../src/resolvers/Mutation/rejectAdmin"; -import { - USER_NOT_AUTHORIZED_SUPERADMIN, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; +import { + USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { rejectAdmin as rejectAdminResolver } from "../../../src/resolvers/Mutation/rejectAdmin"; import type { TestUserType } from "../../helpers/user"; import { createTestUserFunc } from "../../helpers/user"; @@ -56,17 +56,10 @@ describe("resolvers -> Mutation -> rejectAdmin", () => { const context = { userId: testUser1?.id, }; - await User.findByIdAndUpdate( - { - _id: testUser1?._id, - }, - { - userType: "USER", - } - ); + await rejectAdminResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE}` ); @@ -93,9 +86,9 @@ describe("resolvers -> Mutation -> rejectAdmin", () => { ); await rejectAdminResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -105,13 +98,13 @@ describe("resolvers -> Mutation -> rejectAdmin", () => { .spyOn(requestContext, "translate") .mockImplementationOnce((message) => message); try { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser1?._id, + userId: testUser1?._id, }, { $set: { - userType: "SUPERADMIN", + isSuperAdmin: true, }, } ); @@ -128,9 +121,9 @@ describe("resolvers -> Mutation -> rejectAdmin", () => { "../../../src/resolvers/Mutation/rejectAdmin" ); await rejectAdminResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -151,4 +144,30 @@ describe("resolvers -> Mutation -> rejectAdmin", () => { expect(flag).toBe(true); }); + it("throws an errorr if the user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + await AppUserProfile.deleteOne({ + userId: testUser1?._id, + }); + const args: MutationRejectAdminArgs = { + id: testUser2?.id, + }; + const context = { + userId: testUser1?.id, + }; + const { rejectAdmin: rejectAdminResolver } = await import( + "../../../src/resolvers/Mutation/rejectAdmin" + ); + try { + await rejectAdminResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/rejectMembershipRequest.spec.ts b/tests/resolvers/Mutation/rejectMembershipRequest.spec.ts index 95f6536842..56dd0d4895 100644 --- a/tests/resolvers/Mutation/rejectMembershipRequest.spec.ts +++ b/tests/resolvers/Mutation/rejectMembershipRequest.spec.ts @@ -70,9 +70,11 @@ describe("resolvers -> Mutation -> rejectMembershipRequest", () => { await import("../../../src/resolvers/Mutation/rejectMembershipRequest"); await rejectMembershipRequestResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(MEMBERSHIP_REQUEST_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(MEMBERSHIP_REQUEST_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + MEMBERSHIP_REQUEST_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -106,9 +108,11 @@ describe("resolvers -> Mutation -> rejectMembershipRequest", () => { await import("../../../src/resolvers/Mutation/rejectMembershipRequest"); await rejectMembershipRequestResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -153,9 +157,9 @@ describe("resolvers -> Mutation -> rejectMembershipRequest", () => { await import("../../../src/resolvers/Mutation/rejectMembershipRequest"); await rejectMembershipRequestResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -190,19 +194,8 @@ describe("resolvers -> Mutation -> rejectMembershipRequest", () => { } ); - await User.findByIdAndUpdate( - { - _id: testUser?._id, - }, - { - $set: { - userType: "USER", - }, - } - ); - const args: MutationRejectMembershipRequestArgs = { - membershipRequestId: testMembershipRequest?._id, + membershipRequestId: testMembershipRequest?._id.toString() ?? "", }; const context = { @@ -216,9 +209,9 @@ describe("resolvers -> Mutation -> rejectMembershipRequest", () => { ); await rejectMembershipRequestResolverAdminError?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ADMIN.MESSAGE}` ); } diff --git a/tests/resolvers/Mutation/removeActionItem.spec.ts b/tests/resolvers/Mutation/removeActionItem.spec.ts index 780980096a..2f2b5e8b5c 100644 --- a/tests/resolvers/Mutation/removeActionItem.spec.ts +++ b/tests/resolvers/Mutation/removeActionItem.spec.ts @@ -1,34 +1,34 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import type { MutationRemoveActionItemArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { - USER_NOT_FOUND_ERROR, ACTION_ITEM_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, EVENT_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { - createTestUser, - createTestUserAndOrganization, -} from "../../helpers/userAndOrg"; import { removeActionItem as removeActionItemResolver } from "../../../src/resolvers/Mutation/removeActionItem"; +import type { MutationRemoveActionItemArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; +import { + createTestUser, + createTestUserAndOrganization, +} from "../../helpers/userAndOrg"; -import type { TestActionItemCategoryType } from "../../helpers/actionItemCategory"; -import { ActionItem, Event, User } from "../../../src/models"; +import { nanoid } from "nanoid"; +import { ActionItem, AppUserProfile, Event } from "../../../src/models"; import type { TestActionItemType } from "../../helpers/actionItem"; import { createNewTestActionItem, createTestActionItem, } from "../../helpers/actionItem"; +import type { TestActionItemCategoryType } from "../../helpers/actionItemCategory"; import type { TestEventType } from "../../helpers/events"; -import { nanoid } from "nanoid"; let randomUser: TestUserType; let assignedTestUser: TestUserType; @@ -83,8 +83,8 @@ describe("resolvers -> Mutation -> removeActionItem", () => { }; await removeActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -99,8 +99,10 @@ describe("resolvers -> Mutation -> removeActionItem", () => { }; await removeActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ACTION_ITEM_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ACTION_ITEM_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -115,8 +117,10 @@ describe("resolvers -> Mutation -> removeActionItem", () => { }; await removeActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); @@ -150,12 +154,12 @@ describe("resolvers -> Mutation -> removeActionItem", () => { actionItemCategoryId: testCategory?._id, }); - const superAdminTestUser = await User.findOneAndUpdate( + const superAdminTestUser = await AppUserProfile.findOneAndUpdate( { - _id: randomUser?._id, + userId: randomUser?._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, }, { new: true, @@ -167,7 +171,7 @@ describe("resolvers -> Mutation -> removeActionItem", () => { }; const context = { - userId: superAdminTestUser?._id, + userId: superAdminTestUser?.userId, }; const removedActionItemPayload = await removeActionItemResolver?.( @@ -212,8 +216,8 @@ describe("resolvers -> Mutation -> removeActionItem", () => { }; await removeActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -256,4 +260,22 @@ describe("resolvers -> Mutation -> removeActionItem", () => { }) ); }); + it("throws an error if user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const args: MutationRemoveActionItemArgs = { + id: testActionItem?._id, + }; + const context = { + userId: testUser?._id, + }; + try { + await removeActionItemResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeAdmin.spec.ts b/tests/resolvers/Mutation/removeAdmin.spec.ts index b605942cbb..f2dfeb9728 100644 --- a/tests/resolvers/Mutation/removeAdmin.spec.ts +++ b/tests/resolvers/Mutation/removeAdmin.spec.ts @@ -1,26 +1,29 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Organization } from "../../../src/models"; +import { AppUserProfile, Organization, User } from "../../../src/models"; import type { MutationRemoveAdminArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { removeAdmin as removeAdminResolver } from "../../../src/resolvers/Mutation/removeAdmin"; +import { nanoid } from "nanoid"; import { - ORGANIZATION_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_SUPERADMIN, - USER_NOT_FOUND_ERROR, - USER_NOT_ORGANIZATION_ADMIN, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; +import { + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, + USER_NOT_ORGANIZATION_ADMIN, +} from "../../../src/constants"; +import { removeAdmin as removeAdminResolver } from "../../../src/resolvers/Mutation/removeAdmin"; +import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; import type { TestOrganizationType, TestUserType, @@ -29,7 +32,6 @@ import { createTestUser, createTestUserAndOrganization, } from "../../helpers/userAndOrg"; -import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; let MONGOOSE_INSTANCE: typeof mongoose; let testUserRemoved: TestUserType; @@ -63,12 +65,12 @@ describe("resolvers -> Mutation -> removeAdmin", () => { const args: MutationRemoveAdminArgs = { data: { organizationId: Types.ObjectId().toString(), - userId: "", + userId: Types.ObjectId().toString(), }, }; const context = { - userId: testUserRemover?.id, + userId: "", }; const { removeAdmin: removeAdminResolver } = await import( @@ -76,9 +78,38 @@ describe("resolvers -> Mutation -> removeAdmin", () => { ); await removeAdminResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); + } + }); + it("throws NotFoundError if no user exists with _id === args.data.userId", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + try { + const args: MutationRemoveAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: Types.ObjectId().toString(), + }, + }; + + const context = { + userId: testUserRemover?.id, + }; + + const { removeAdmin: removeAdminResolver } = await import( + "../../../src/resolvers/Mutation/removeAdmin" + ); + + await removeAdminResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -104,9 +135,101 @@ describe("resolvers -> Mutation -> removeAdmin", () => { ); await removeAdminResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } + }); + it(`throws user not authorized error if user does not have appUseProfile`, async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + try { + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const args: MutationRemoveAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: newUser._id, + }, + }; + const context = { + userId: testUserRemover?.id, + }; + const { removeAdmin: removeAdminResolver } = await import( + "../../../src/resolvers/Mutation/removeAdmin" + ); + await removeAdminResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); + it(`throws user not authorized error if current user does not have appUseProfile`, async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + try { + const args: MutationRemoveAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: testUserRemoved?.id, + }, + }; + const newUser = await User.create({ + email: `email${nanoid().toLowerCase()}@gmail.com`, + password: `pass${nanoid().toLowerCase()}`, + firstName: `firstName${nanoid().toLowerCase()}`, + lastName: `lastName${nanoid().toLowerCase()}`, + image: null, + }); + const context = { + userId: newUser?.id, + }; + const { removeAdmin: removeAdminResolver } = await import( + "../../../src/resolvers/Mutation/removeAdmin" + ); + await removeAdminResolver?.({}, args, context); + } catch (error: unknown) { + // console.log(error); + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); + it("throws error if no user found", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + try { + const args: MutationRemoveAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: Types.ObjectId().toString(), + }, + }; + const context = { + userId: testUserRemover?.id, + }; + const { removeAdmin: removeAdminResolver } = await import( + "../../../src/resolvers/Mutation/removeAdmin" + ); + await removeAdminResolver?.({}, args, context); + } catch (error: unknown) { + console.log(error); + expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -152,9 +275,9 @@ describe("resolvers -> Mutation -> removeAdmin", () => { ); await removeAdminAdminError?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_ORGANIZATION_ADMIN.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_ORGANIZATION_ADMIN.MESSAGE}` ); } @@ -204,11 +327,11 @@ describe("resolvers -> Mutation -> removeAdmin", () => { ); await removeAdminAdminError?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith( USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE ); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE}` ); } @@ -234,7 +357,16 @@ describe("resolvers -> Mutation -> removeAdmin", () => { { $set: { adminApproved: true, - userType: "SUPERADMIN", + }, + } + ); + await AppUserProfile.updateOne( + { + userId: testUserRemover?.id, + }, + { + $set: { + isSuperAdmin: true, }, } ); @@ -252,12 +384,27 @@ describe("resolvers -> Mutation -> removeAdmin", () => { const removeAdminPayload = await removeAdminResolver?.({}, args, context); - const updatedTestUser = await User.findOne({ - _id: testUserRemoved?._id, - }) - .select(["-password"]) - .lean(); + const updatedTestUser = await AppUserProfile.findOne({ + userId: testUserRemoved?._id, + }).lean(); expect(removeAdminPayload).toEqual(updatedTestUser); }); + it("throws error if user does not exists", async () => { + const context = { + userId: Types.ObjectId().toString(), + }; + + const args: MutationRemoveAdminArgs = { + data: { + organizationId: testOrganization?.id, + userId: testUserRemover?.id, + }, + }; + try { + await removeAdminResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } + }); }); diff --git a/tests/resolvers/Mutation/removeAdvertisement.spec.ts b/tests/resolvers/Mutation/removeAdvertisement.spec.ts index 79a05d8523..c07e952092 100644 --- a/tests/resolvers/Mutation/removeAdvertisement.spec.ts +++ b/tests/resolvers/Mutation/removeAdvertisement.spec.ts @@ -88,13 +88,21 @@ describe("resolvers -> Mutation -> removeAdvertisement", () => { context ); + const removeAdvertisementPayloadFalsyId = await removeAdvertisement?.( + {}, + { id: "" }, + context + ); expect(removeAdvertisementPayload).toHaveProperty( "_id", createdAdvertisementId ); - + if (removeAdvertisementPayloadFalsyId) { + expect(removeAdvertisementPayloadFalsyId).toHaveProperty("_id", ""); + } else { + console.error("removeAdvertisementPayloadFalsyId is undefined or null"); + } expect(removeAdvertisementPayload).toHaveProperty("name", "myad"); - expect(removeAdvertisementPayload).toHaveProperty( "link", "https://www.example.com" @@ -122,11 +130,13 @@ describe("resolvers -> Mutation -> removeAdvertisement", () => { { id: "64d1f8cb77a4b51004f824b8" }, context ); - } catch (error: any) { - expect(spy).toBeCalledWith(ADVERTISEMENT_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( - `Translated ${ADVERTISEMENT_NOT_FOUND_ERROR.MESSAGE}` - ); + } catch (error: unknown) { + if (error instanceof Error) { + expect(spy).toBeCalledWith(ADVERTISEMENT_NOT_FOUND_ERROR.MESSAGE); + expect(error.message).toEqual( + `Translated ${ADVERTISEMENT_NOT_FOUND_ERROR.MESSAGE}` + ); + } } }); }); diff --git a/tests/resolvers/Mutation/removeComment.spec.ts b/tests/resolvers/Mutation/removeComment.spec.ts index c6b5a216ef..1d5bf70442 100644 --- a/tests/resolvers/Mutation/removeComment.spec.ts +++ b/tests/resolvers/Mutation/removeComment.spec.ts @@ -1,36 +1,36 @@ import "dotenv/config"; -import type { Document } from "mongoose"; import type mongoose from "mongoose"; +import type { Document } from "mongoose"; import { Types } from "mongoose"; -import type { InterfaceComment } from "../../../src/models"; -import { Comment, Post, User } from "../../../src/models"; -import type { MutationRemoveCommentArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; -import { removeComment as removeCommentResolver } from "../../../src/resolvers/Mutation/removeComment"; import { - COMMENT_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; +import { + COMMENT_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import type { InterfaceComment } from "../../../src/models"; +import { AppUserProfile, Comment, Post, User } from "../../../src/models"; +import { removeComment as removeCommentResolver } from "../../../src/resolvers/Mutation/removeComment"; +import { cacheComments } from "../../../src/services/CommentCache/cacheComments"; +import type { MutationRemoveCommentArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import type { TestPostType } from "../../helpers/posts"; import { createTestPost } from "../../helpers/posts"; -import { cacheComments } from "../../../src/services/CommentCache/cacheComments"; +import type { TestUserType } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; let testPost: TestPostType; let testComment: - | (InterfaceComment & Document) + | (InterfaceComment & Document) | null; beforeAll(async () => { @@ -89,9 +89,9 @@ describe("resolvers -> Mutation -> removeComment", () => { ); await removeCommentResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -114,9 +114,9 @@ describe("resolvers -> Mutation -> removeComment", () => { ); await removeCommentResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(COMMENT_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(COMMENT_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(COMMENT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -170,9 +170,11 @@ describe("resolvers -> Mutation -> removeComment", () => { ); await removeCommentResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); @@ -184,7 +186,7 @@ describe("resolvers -> Mutation -> removeComment", () => { }, { $set: { - creatorId: testUser!._id, + creatorId: testUser?._id, }, }, { @@ -237,4 +239,31 @@ describe("resolvers -> Mutation -> removeComment", () => { expect(commentExists).toBeFalsy(); expect(testUpdatedPost?.commentCount).toEqual(0); }); + it("throws an error if the user does not have AppUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const args: MutationRemoveCommentArgs = { + id: testComment?._id.toString() ?? "", + }; + const context = { + userId: testUser?._id, + }; + try { + const { removeComment: removeCommentResolver } = await import( + "../../../src/resolvers/Mutation/removeComment" + ); + + await removeCommentResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeEvent.spec.ts b/tests/resolvers/Mutation/removeEvent.spec.ts index ae83acaa73..8ba8a0c279 100644 --- a/tests/resolvers/Mutation/removeEvent.spec.ts +++ b/tests/resolvers/Mutation/removeEvent.spec.ts @@ -1,29 +1,32 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Event, ActionItem } from "../../../src/models"; +import { ActionItem, AppUserProfile, Event } from "../../../src/models"; import type { MutationRemoveEventArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { removeEvent as removeEventResolver } from "../../../src/resolvers/Mutation/removeEvent"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { EVENT_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; +import { removeEvent as removeEventResolver } from "../../../src/resolvers/Mutation/removeEvent"; +import { cacheEvents } from "../../../src/services/EventCache/cacheEvents"; +import { createTestActionItems } from "../../helpers/actionItem"; +import type { TestEventType } from "../../helpers/events"; +import { createTestEvent } from "../../helpers/events"; import type { + // TestAppUserProfileType, TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; -import type { TestEventType } from "../../helpers/events"; -import { createTestEvent } from "../../helpers/events"; -import { cacheEvents } from "../../../src/services/EventCache/cacheEvents"; -import { createTestActionItems } from "../../helpers/actionItem"; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; +// let testUserAppProfile: TestAppUserProfileType; let newTestUser: TestUserType; +// let newTestUserAppProfile: TestAppUserProfileType; let testOrganization: TestOrganizationType; let testEvent: TestEventType; let newTestEvent: TestEventType; @@ -60,9 +63,9 @@ describe("resolvers -> Mutation -> removeEvent", () => { ); await removeEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -85,9 +88,9 @@ describe("resolvers -> Mutation -> removeEvent", () => { ); await removeEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -99,9 +102,9 @@ describe("resolvers -> Mutation -> removeEvent", () => { .spyOn(requestContext, "translate") .mockImplementationOnce((message) => message); try { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $set: { @@ -134,16 +137,18 @@ describe("resolvers -> Mutation -> removeEvent", () => { ); await removeEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); it(`removes event with _id === args.id and returns it`, async () => { - await User.updateOne( + await AppUserProfile.updateOne( { - _id: testUser?._id, + userId: testUser?._id, }, { $push: { @@ -185,14 +190,15 @@ describe("resolvers -> Mutation -> removeEvent", () => { updatedAt: expect.anything(), }); - const updatedTestUser = await User.findOne({ - _id: testUser?._id, + const updatedTestUserAppProfile = await AppUserProfile.findOne({ + userId: testUser?._id, }) + .select(["createdEvents", "eventAdmin"]) .lean(); - expect(updatedTestUser?.createdEvents).toEqual([]); - expect(updatedTestUser?.eventAdmin).toEqual([]); + expect(updatedTestUserAppProfile?.createdEvents).toEqual([]); + expect(updatedTestUserAppProfile?.eventAdmin).toEqual([]); const updatedTestEvent = await Event.findOne({ _id: testEvent?._id, @@ -224,4 +230,30 @@ describe("resolvers -> Mutation -> removeEvent", () => { expect(deletedActionItems).toEqual([]); }); + it("throws an error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const args: MutationRemoveEventArgs = { + id: testEvent?.id, + }; + const context = { + userId: testUser?._id, + }; + try { + const { removeEvent: removeEventResolver } = await import( + "../../../src/resolvers/Mutation/removeEvent" + ); + await removeEventResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeEventAttendee.spec.ts b/tests/resolvers/Mutation/removeEventAttendee.spec.ts index 1ed20abac6..9bdb026de0 100644 --- a/tests/resolvers/Mutation/removeEventAttendee.spec.ts +++ b/tests/resolvers/Mutation/removeEventAttendee.spec.ts @@ -1,18 +1,18 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { EventAttendee, User } from "../../../src/models"; -import type { MutationRemoveEventAttendeeArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { EVENT_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, USER_NOT_REGISTERED_FOR_EVENT, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { createTestUser, type TestUserType } from "../../helpers/userAndOrg"; +import { AppUserProfile, EventAttendee, User } from "../../../src/models"; +import type { MutationRemoveEventAttendeeArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import { createTestEvent, type TestEventType } from "../../helpers/events"; +import { createTestUser, type TestUserType } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; @@ -52,8 +52,8 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { ); await removeEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -75,15 +75,15 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { }, }; - const context = { userId: randomTestUser!._id }; + const context = { userId: randomTestUser?._id }; const { removeEventAttendee: removeEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/removeEventAttendee" ); await removeEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${EVENT_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); @@ -101,19 +101,19 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { const args: MutationRemoveEventAttendeeArgs = { data: { userId: Types.ObjectId().toString(), - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: randomTestUser!._id }; + const context = { userId: randomTestUser?._id }; const { removeEventAttendee: removeEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/removeEventAttendee" ); await removeEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); @@ -131,19 +131,19 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { const args: MutationRemoveEventAttendeeArgs = { data: { userId: Types.ObjectId().toString(), - eventId: testEvent!._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { removeEventAttendee: removeEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/removeEventAttendee" ); await removeEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -160,20 +160,20 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { try { const args: MutationRemoveEventAttendeeArgs = { data: { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; const { removeEventAttendee: removeEventAttendeeResolver } = await import( "../../../src/resolvers/Mutation/removeEventAttendee" ); await removeEventAttendeeResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_REGISTERED_FOR_EVENT.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -185,12 +185,12 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { it(`unregisters the request user for the event successfully and returns the request user`, async () => { const args: MutationRemoveEventAttendeeArgs = { data: { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }, }; - const context = { userId: testUser!._id }; + const context = { userId: testUser?._id }; await EventAttendee.create({ ...args.data }); @@ -201,7 +201,7 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { const payload = await removeEventAttendeeResolver?.({}, args, context); const requestUser = await User.findOne({ - _id: testUser!._id, + _id: testUser?._id, }).lean(); const isUserRegistered = await EventAttendee.exists({ @@ -211,4 +211,33 @@ describe("resolvers -> Mutation -> removeEventAttendee", () => { expect(payload).toEqual(requestUser); expect(isUserRegistered).toBeFalsy(); }); + it("throws error if the user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + const args: MutationRemoveEventAttendeeArgs = { + data: { + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", + }, + }; + const context = { + userId: testUser?._id, + }; + try { + const { removeEventAttendee: removeEventAttendeeResolver } = await import( + "../../../src/resolvers/Mutation/removeEventAttendee" + ); + await removeEventAttendeeResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeOrganization.spec.ts b/tests/resolvers/Mutation/removeOrganization.spec.ts index 32e27e4cfe..e3718e1108 100644 --- a/tests/resolvers/Mutation/removeOrganization.spec.ts +++ b/tests/resolvers/Mutation/removeOrganization.spec.ts @@ -1,44 +1,46 @@ import "dotenv/config"; -import type { Document } from "mongoose"; import type mongoose from "mongoose"; +import type { Document } from "mongoose"; import { Types } from "mongoose"; import type { - InterfaceOrganization, + InterfaceActionItem, + InterfaceActionItemCategory, InterfaceComment, + InterfaceOrganization, InterfacePost, - InterfaceActionItemCategory, - InterfaceActionItem, } from "../../../src/models"; import { - User, - Organization, - Post, + ActionItem, + ActionItemCategory, + AppUserProfile, Comment, MembershipRequest, - ActionItemCategory, - ActionItem, + Organization, + Post, + User, } from "../../../src/models"; import type { MutationRemoveOrganizationArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { removeOrganization as removeOrganizationResolver } from "../../../src/resolvers/Mutation/removeOrganization"; import { - ORGANIZATION_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_SUPERADMIN, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; +import { + ORGANIZATION_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_AUTHORIZED_SUPERADMIN, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { removeOrganization as removeOrganizationResolver } from "../../../src/resolvers/Mutation/removeOrganization"; +import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; import { createTestUserFunc } from "../../helpers/user"; import type { TestUserType } from "../../helpers/userAndOrg"; -import { cacheOrganizations } from "../../../src/services/OrganizationCache/cacheOrganizations"; /* eslint-disable */ let MONGOOSE_INSTANCE: typeof mongoose; let testUsers: TestUserType[]; @@ -67,7 +69,7 @@ beforeAll(async () => { sortingCode: "ABC-123", state: "Delhi", }, - isPublic: true, + userRegistrationRequired: true, creatorId: testUsers[0]?._id, admins: [testUsers[0]?._id], members: [testUsers[1]?._id], @@ -81,13 +83,22 @@ beforeAll(async () => { }, { $set: { - createdOrganizations: [testOrganization._id], - adminFor: [testOrganization._id], joinedOrganizations: [testOrganization._id], organizationsBlockedBy: [testOrganization._id], }, } ); + await AppUserProfile.updateOne( + { + user: testUsers[0]?._id, + }, + { + $set: { + createdOrganizations: [testOrganization._id], + adminFor: [testOrganization._id], + }, + } + ); await User.updateOne( { @@ -303,10 +314,17 @@ describe("resolvers -> Mutation -> removeOrganization", () => { { $set: { adminApproved: true, - userType: "SUPERADMIN", }, } ); + await AppUserProfile.updateOne( + { + userId: testUsers[0]?._id, + }, + { + isSuperAdmin: true, + } + ); const args: MutationRemoveOrganizationArgs = { id: testOrganization._id, @@ -328,7 +346,7 @@ describe("resolvers -> Mutation -> removeOrganization", () => { .select(["-password"]) .lean(); - expect(removeOrganizationPayload).toEqual(updatedTestUser); + expect(removeOrganizationPayload?.user).toEqual(updatedTestUser); const updatedTestUser1 = await User.findOne({ _id: testUsers[1]?._id, @@ -382,7 +400,7 @@ describe("resolvers -> Mutation -> removeOrganization", () => { sortingCode: "ABC-123", state: "Delhi", }, - isPublic: true, + userRegistrationRequired: true, creatorId: testUsers[0]?._id, admins: [testUsers[0]?._id], members: [testUsers[1]?._id], @@ -424,10 +442,39 @@ describe("resolvers -> Mutation -> removeOrganization", () => { context ); - expect(removeOrganizationPayload).toEqual({ + expect(removeOrganizationPayload?.user).toEqual({ ...updatedTestUser, updatedAt: expect.anything(), }); expect(deleteImageSpy).toBeCalledWith("images/fake-image-path.png"); }); + it(`throws error if user does not have appUserProfile`, async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementation((message) => `Translated ${message}`); + await AppUserProfile.deleteOne({ + userId: testUsers[0]?._id, + }); + try { + const args: MutationRemoveOrganizationArgs = { + id: "", + }; + + const context = { + userId: testUsers[0]?._id, + }; + + const { removeOrganization: removeOrganizationResolver } = await import( + "../../../src/resolvers/Mutation/removeOrganization" + ); + + await removeOrganizationResolver?.({}, args, context); + } catch (error: any) { + expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect(error.message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeOrganizationCustomField.spec.ts b/tests/resolvers/Mutation/removeOrganizationCustomField.spec.ts index b40399cc27..f8a8be3153 100644 --- a/tests/resolvers/Mutation/removeOrganizationCustomField.spec.ts +++ b/tests/resolvers/Mutation/removeOrganizationCustomField.spec.ts @@ -1,14 +1,14 @@ -import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; -import { Types } from "mongoose"; import type mongoose from "mongoose"; -import { removeOrganizationCustomField } from "../../../src/resolvers/Mutation/removeOrganizationCustomField"; +import { Types } from "mongoose"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { addOrganizationCustomField } from "../../../src/resolvers/Mutation/addOrganizationCustomField"; +import { removeOrganizationCustomField } from "../../../src/resolvers/Mutation/removeOrganizationCustomField"; +import { connect, disconnect } from "../../helpers/db"; import { createTestUserAndOrganization, type TestOrganizationType, type TestUserType, } from "../../helpers/userAndOrg"; -import { connect, disconnect } from "../../helpers/db"; import { CUSTOM_FIELD_NOT_FOUND, @@ -19,7 +19,7 @@ import { import { createTestUser } from "../../helpers/userAndOrg"; -import { OrganizationCustomField } from "../../../src/models"; +import { AppUserProfile, OrganizationCustomField } from "../../../src/models"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -38,11 +38,6 @@ afterAll(async () => { describe("resolvers => Mutation => removeOrganizationCustomField", () => { it("should remove field added by the organization", async () => { - const { requestContext } = await import("../../../src/libraries"); - const spy = vi - .spyOn(requestContext, "translate") - .mockImplementationOnce((message) => `Translated ${message}`); - const customField = await addOrganizationCustomField?.( {}, { @@ -102,10 +97,6 @@ describe("resolvers => Mutation => removeOrganizationCustomField", () => { } ); - const initialCustomFields = await OrganizationCustomField.find({ - organizationId: testOrganization?._id, - }); - expect(customField).toBeDefined(); expect(customField?.organizationId.toString()).toBe( testOrganization?._id.toString() @@ -118,9 +109,9 @@ describe("resolvers => Mutation => removeOrganizationCustomField", () => { }; try { await removeOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); } @@ -142,9 +133,9 @@ describe("resolvers => Mutation => removeOrganizationCustomField", () => { try { await removeOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(CUSTOM_FIELD_NOT_FOUND.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${CUSTOM_FIELD_NOT_FOUND.MESSAGE}` ); } @@ -181,15 +172,19 @@ describe("resolvers => Mutation => removeOrganizationCustomField", () => { try { await removeOrganizationCustomField?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } }); it("should throw an error when organization is not found", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); const customField = await addOrganizationCustomField?.( {}, { @@ -215,10 +210,36 @@ describe("resolvers => Mutation => removeOrganizationCustomField", () => { try { await removeOrganizationCustomField?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect(spy).toHaveBeenLastCalledWith( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); + expect((error as Error).message).toEqual( `Translated ${ORGANIZATION_NOT_FOUND_ERROR.MESSAGE}` ); } }); + it("throws an error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?._id, + }); + const context = { userId: newUser?._id }; + const args = { + organizationId: testOrganization?._id as string, + customFieldId: Types.ObjectId().toString() as string, + }; + try { + await removeOrganizationCustomField?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removePost.spec.ts b/tests/resolvers/Mutation/removePost.spec.ts index 9a2fb19d20..9391e6bb8d 100644 --- a/tests/resolvers/Mutation/removePost.spec.ts +++ b/tests/resolvers/Mutation/removePost.spec.ts @@ -4,25 +4,25 @@ import { Types } from "mongoose"; import type { MutationRemovePostArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; +import { + afterAll, + afterEach, + beforeAll, + describe, + expect, + it, + vi, +} from "vitest"; import { POST_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import type { TestUserType } from "../../helpers/userAndOrg"; -import { createTestUser } from "../../helpers/userAndOrg"; +import { AppUserProfile, Post } from "../../../src/models"; import type { TestPostType } from "../../helpers/posts"; import { createTestPost } from "../../helpers/posts"; -import { - beforeAll, - afterAll, - describe, - it, - expect, - vi, - afterEach, -} from "vitest"; -import { Post } from "../../../src/models"; +import type { TestUserType } from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; @@ -66,8 +66,8 @@ describe("resolvers -> Mutation -> removePost", () => { ); await removePostResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -93,8 +93,8 @@ describe("resolvers -> Mutation -> removePost", () => { ); await removePostResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${POST_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -120,8 +120,8 @@ describe("resolvers -> Mutation -> removePost", () => { ); await removePostResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); } @@ -234,4 +234,34 @@ describe("resolvers -> Mutation -> removePost", () => { expect(removePostPayload).toEqual(updatedPost); expect(deleteVideoSpy).toBeCalledWith("videos/fakeVideoPathvideo.png"); }); + it("throws an error if the user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + vi.spyOn(requestContext, "translate").mockImplementationOnce( + (message) => `Translated ${message}` + ); + + try { + const args: MutationRemovePostArgs = { + id: testPost?.id, + }; + + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const context = { + userId: newUser?.id, + }; + + const { removePost: removePostResolver } = await import( + "../../../src/resolvers/Mutation/removePost" + ); + + await removePostResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeSampleOrganization.spec.ts b/tests/resolvers/Mutation/removeSampleOrganization.spec.ts index cc5030b597..fca79c973e 100644 --- a/tests/resolvers/Mutation/removeSampleOrganization.spec.ts +++ b/tests/resolvers/Mutation/removeSampleOrganization.spec.ts @@ -1,16 +1,17 @@ -import type { InterfaceOrganization } from "../../../src/models"; -import { SampleData, Organization } from "../../../src/models"; -import { generateUserData } from "../../../src/utilities/createSampleOrganizationUtil"; -import { expect, describe, it, vi, afterAll, beforeAll } from "vitest"; -import { removeSampleOrganization } from "../../../src/resolvers/Mutation/removeSampleOrganization"; -import type mongoose from "mongoose"; import { faker } from "@faker-js/faker"; -import { connect, disconnect } from "../../helpers/db"; +import type mongoose from "mongoose"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; +import type { InterfaceOrganization } from "../../../src/models"; +import { AppUserProfile, Organization, SampleData } from "../../../src/models"; +import { removeSampleOrganization } from "../../../src/resolvers/Mutation/removeSampleOrganization"; +import { generateUserData } from "../../../src/utilities/createSampleOrganizationUtil"; +import { connect, disconnect } from "../../helpers/db"; +import { createTestUser } from "../../helpers/userAndOrg"; /* eslint-disable */ const ORGANIZATION_ID = ((): InterfaceOrganization & mongoose.Document => { @@ -62,11 +63,14 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => (message) => message ); - const admin = generateUserData(ORGANIZATION_ID.toString(), "ADMIN"); - (await admin).save(); + const userData = await generateUserData( + ORGANIZATION_ID.toString(), + "ADMIN" + ); + const admin = userData.user; const args = {}; - const adminContext = { userId: (await admin)._id }; + const adminContext = { userId: admin._id }; const parent = {}; const adminResult = await removeSampleOrganization!( @@ -89,7 +93,7 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => _id, name: faker.company.name(), description: faker.lorem.sentences(), - isPublic: true, + userRegistrationRequired: false, creatorId: creatorId, status: "ACTIVE", members: [creatorId], @@ -111,15 +115,14 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => }); sampleDocument.save(); - const superadmin = generateUserData( + const userData = await generateUserData( organization._id.toString(), "SUPERADMIN" ); - - (await superadmin).save(); + const superAdmin = userData.user; const args = {}; - const superAdminContext = { userId: (await superadmin)._id }; + const superAdminContext = { userId: superAdmin._id }; const parent = {}; const superAdminResult = await removeSampleOrganization!( @@ -143,7 +146,7 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => _id, name: faker.company.name(), description: faker.lorem.sentences(), - isPublic: true, + userRegistrationRequired: false, creatorId: creatorId, status: "ACTIVE", members: [creatorId], @@ -165,10 +168,14 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => }); sampleModal.save(); - const newUser = generateUserData(organization._id.toString(), "USER"); + const userData = await generateUserData( + organization._id.toString(), + "USER" + ); + const newUser = userData.user; const args = {}; - const context = { userId: (await newUser)._id }; + const context = { userId: newUser._id }; const parent = {}; try { @@ -197,21 +204,24 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => } }); - it("should NOT throw error when user is ADMIN", async () => { + it("should NOT throw error when organization doesn't exist", async () => { const { requestContext } = await import("../../../src/libraries"); vi.spyOn(requestContext, "translate").mockImplementation( (message) => message ); - const randomOrganizationId = faker.database.mongodbObjectId(); - - const admin = generateUserData(randomOrganizationId, "ADMIN"); - (await admin).save(); + const userData = await generateUserData( + ORGANIZATION_ID.toString(), + "ADMIN" + ); + const admin = userData.user; const args = {}; - const adminContext = { userId: (await admin)._id }; + const adminContext = { userId: admin._id }; const parent = {}; + await SampleData.deleteMany({ collectionName: "Organization" }); + try { await removeSampleOrganization!(parent, args, adminContext); } catch (error: any) { @@ -219,21 +229,22 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => } }); - it("should NOT throw error when organization doesn't exist", async () => { + it("should throw error when the collection name is not a valid one", async () => { const { requestContext } = await import("../../../src/libraries"); vi.spyOn(requestContext, "translate").mockImplementation( (message) => message ); - const admin = generateUserData(ORGANIZATION_ID.toString(), "ADMIN"); - (await admin).save(); + const userData = await generateUserData( + ORGANIZATION_ID.toString(), + "ADMIN" + ); + const admin = userData.user; const args = {}; - const adminContext = { userId: (await admin)._id }; + const adminContext = { userId: admin._id }; const parent = {}; - await SampleData.deleteMany({ collectionName: "Organization" }); - try { await removeSampleOrganization!(parent, args, adminContext); } catch (error: any) { @@ -241,23 +252,25 @@ describe("Remove Sample Organization Resolver - User Authorization", async () => } }); - it("should throw error when the collection name is not a valid one", async () => { + it("should throw user not found error when user is non-existent", async () => { const { requestContext } = await import("../../../src/libraries"); vi.spyOn(requestContext, "translate").mockImplementation( (message) => message ); - const admin = generateUserData(ORGANIZATION_ID.toString(), "ADMIN"); - (await admin).save(); + const testUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: testUser?.id, + }); const args = {}; - const adminContext = { userId: (await admin)._id }; + const adminContext = { userId: testUser?._id }; const parent = {}; try { await removeSampleOrganization!(parent, args, adminContext); } catch (error: any) { - expect(error.message).toBe(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + expect(error.message).toBe(USER_NOT_AUTHORIZED_ERROR.MESSAGE); } }); }); diff --git a/tests/resolvers/Mutation/removeUserCustomData.spec.ts b/tests/resolvers/Mutation/removeUserCustomData.spec.ts index 639b6e01c3..d398cd2928 100644 --- a/tests/resolvers/Mutation/removeUserCustomData.spec.ts +++ b/tests/resolvers/Mutation/removeUserCustomData.spec.ts @@ -1,8 +1,9 @@ -import mongoose, { Types } from "mongoose"; -import { describe, it, expect, beforeAll, afterAll, vi } from "vitest"; +import mongoose from "mongoose"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { addUserCustomData } from "../../../src/resolvers/Mutation/addUserCustomData"; import { removeUserCustomData } from "../../../src/resolvers/Mutation/removeUserCustomData"; +import { connect, disconnect } from "../../helpers/db"; import type { TestOrganizationType, TestUserType, @@ -11,7 +12,6 @@ import { createTestUser, createTestUserAndOrganization, } from "../../helpers/userAndOrg"; -import { connect, disconnect } from "../../helpers/db"; import { CUSTOM_DATA_NOT_FOUND, @@ -19,6 +19,7 @@ import { USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, } from "../../../src/constants"; +import { AppUserProfile } from "../../../src/models"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -95,9 +96,9 @@ describe("removeUserCustomData mutation", () => { try { await removeUserCustomData?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); } @@ -130,9 +131,9 @@ describe("removeUserCustomData mutation", () => { try { await removeUserCustomData?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -165,11 +166,11 @@ describe("removeUserCustomData mutation", () => { try { await removeUserCustomData?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith( ORGANIZATION_NOT_FOUND_ERROR.MESSAGE ); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${ORGANIZATION_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -202,11 +203,11 @@ describe("removeUserCustomData mutation", () => { try { await removeUserCustomData?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith( ORGANIZATION_NOT_FOUND_ERROR.MESSAGE ); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${ORGANIZATION_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -226,11 +227,32 @@ describe("removeUserCustomData mutation", () => { try { await removeUserCustomData?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(CUSTOM_DATA_NOT_FOUND.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${CUSTOM_DATA_NOT_FOUND.MESSAGE}` ); } }); + it("throws an error if the user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ userId: testUser?._id }); + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + const args = { + organizationId: testOrganization?._id, + }; + const context = { + userId: testUser?._id, + }; + try { + await removeUserCustomData?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeUserFamily.spec.ts b/tests/resolvers/Mutation/removeUserFamily.spec.ts index c2fbf8cfcd..5fd40b7681 100644 --- a/tests/resolvers/Mutation/removeUserFamily.spec.ts +++ b/tests/resolvers/Mutation/removeUserFamily.spec.ts @@ -6,23 +6,25 @@ import type { MutationRemoveUserFamilyArgs } from "../../../src/types/generatedG import { connect, disconnect } from "../../helpers/db"; import { - USER_FAMILY_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; -import { createTestUserFunc } from "../../helpers/userAndUserFamily"; +import { + USER_FAMILY_NOT_FOUND_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile } from "../../../src/models"; +import { createTestUser } from "../../helpers/userAndOrg"; import type { TestUserFamilyType, TestUserType, } from "../../helpers/userAndUserFamily"; +import { createTestUserFunc } from "../../helpers/userAndUserFamily"; let MONGOOSE_INSTANCE: typeof mongoose; let testUsers: TestUserType[]; @@ -136,4 +138,36 @@ describe("resolvers -> Mutation -> removeUserFamily", () => { ); } }); + it("throws error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementation((message) => message); + + try { + const args: MutationRemoveUserFamilyArgs = { + familyId: testUserFamily?.id, + }; + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + + const context = { + userId: newUser?.id, + }; + + const { removeUserFamily: removeUserFamilyResolver } = await import( + "../../../src/resolvers/Mutation/removeUserFamily" + ); + + await removeUserFamilyResolver?.({}, args, context); + } catch (error) { + // console.log(error); + expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `${USER_NOT_FOUND_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/removeUserFromGroupChat.spec.ts b/tests/resolvers/Mutation/removeUserFromGroupChat.spec.ts index e127f4ee19..cfb6adbf71 100644 --- a/tests/resolvers/Mutation/removeUserFromGroupChat.spec.ts +++ b/tests/resolvers/Mutation/removeUserFromGroupChat.spec.ts @@ -57,9 +57,10 @@ describe("resolvers -> Mutation -> removeUserFromGroupChat", () => { await import("../../../src/resolvers/Mutation/removeUserFromGroupChat"); await removeUserFromGroupChatResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { + console.log((error as Error).message); expect(spy).toBeCalledWith(CHAT_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(CHAT_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual(CHAT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -94,9 +95,11 @@ describe("resolvers -> Mutation -> removeUserFromGroupChat", () => { await import("../../../src/resolvers/Mutation/removeUserFromGroupChat"); await removeUserFromGroupChatResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); @@ -142,9 +145,11 @@ describe("resolvers -> Mutation -> removeUserFromGroupChat", () => { await import("../../../src/resolvers/Mutation/removeUserFromGroupChat"); await removeUserFromGroupChatResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); @@ -196,8 +201,8 @@ describe("resolvers -> Mutation -> removeUserFromGroupChat", () => { const deletedOrgaization = await Organization.findOneAndRemove({ _id: testOrganization?._id, }); - - await deleteOrganizationFromCache(deletedOrgaization!); + if (deletedOrgaization) + await deleteOrganizationFromCache(deletedOrgaization); try { const args: MutationRemoveUserFromGroupChatArgs = { @@ -213,9 +218,11 @@ describe("resolvers -> Mutation -> removeUserFromGroupChat", () => { await import("../../../src/resolvers/Mutation/removeUserFromGroupChat"); await removeUserFromGroupChatResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); }); diff --git a/tests/resolvers/Mutation/removeUserTag.spec.ts b/tests/resolvers/Mutation/removeUserTag.spec.ts index 1fa23c86ff..cf8bca074d 100644 --- a/tests/resolvers/Mutation/removeUserTag.spec.ts +++ b/tests/resolvers/Mutation/removeUserTag.spec.ts @@ -5,24 +5,28 @@ import type { MutationRemoveUserTagArgs } from "../../../src/types/generatedGrap import { connect, disconnect } from "../../helpers/db"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, - TAG_NOT_FOUND, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, - afterEach, + it, vi, } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; -import { createTestUser } from "../../helpers/userAndOrg"; -import { OrganizationTagUser, TagUser } from "../../../src/models"; +import { + TAG_NOT_FOUND, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { + AppUserProfile, + OrganizationTagUser, + TagUser, +} from "../../../src/models"; import type { TestUserTagType } from "../../helpers/tags"; import { createTwoLevelTagsWithOrg } from "../../helpers/tags"; +import type { TestUserType } from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; let testUser: TestUserType; let randomUser: TestUserType; @@ -88,9 +92,9 @@ describe("resolvers -> Mutation -> removeUserTag", () => { ); await removeUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -116,9 +120,11 @@ describe("resolvers -> Mutation -> removeUserTag", () => { ); await removeUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(TAG_NOT_FOUND.MESSAGE); - expect(error.message).toEqual(`Translated ${TAG_NOT_FOUND.MESSAGE}`); + expect((error as Error).message).toEqual( + `Translated ${TAG_NOT_FOUND.MESSAGE}` + ); } }); @@ -142,9 +148,9 @@ describe("resolvers -> Mutation -> removeUserTag", () => { ); await removeUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); } @@ -192,4 +198,33 @@ describe("resolvers -> Mutation -> removeUserTag", () => { expect(userTagExists).toBeFalsy(); }); + it("throws error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + try { + const args: MutationRemoveUserTagArgs = { + id: rootTag ? rootTag._id.toString() : "", + }; + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const context = { + userId: newUser?.id, + }; + + const { removeUserTag: removeUserTagResolver } = await import( + "../../../src/resolvers/Mutation/removeUserTag" + ); + + await removeUserTagResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/revokeRefreshTokenForUser.spec.ts b/tests/resolvers/Mutation/revokeRefreshTokenForUser.spec.ts index 1201ea916d..7447b64004 100644 --- a/tests/resolvers/Mutation/revokeRefreshTokenForUser.spec.ts +++ b/tests/resolvers/Mutation/revokeRefreshTokenForUser.spec.ts @@ -1,9 +1,9 @@ import "dotenv/config"; -import { User } from "../../../src/models"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; +import { AppUserProfile } from "../../../src/models"; import { revokeRefreshTokenForUser as revokeRefreshTokenForUserResolver } from "../../../src/resolvers/Mutation/revokeRefreshTokenForUser"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; +import { connect, disconnect } from "../../helpers/db"; import type { TestUserType } from "../../helpers/user"; import { createTestUserFunc } from "../../helpers/user"; @@ -32,8 +32,8 @@ describe("resolvers -> Mutation -> revokeRefreshTokenForUser", () => { expect(revokeRefreshTokenForUserPayload).toEqual(true); - const testSaveFcmTokenPayload = await User.findOne({ - _id: testUser?._id, + const testSaveFcmTokenPayload = await AppUserProfile.findOne({ + userId: testUser?._id, }) .select("token") .lean(); diff --git a/tests/resolvers/Mutation/saveFcmToken.spec.ts b/tests/resolvers/Mutation/saveFcmToken.spec.ts index 44fdceb1c2..cdf2bcf55a 100644 --- a/tests/resolvers/Mutation/saveFcmToken.spec.ts +++ b/tests/resolvers/Mutation/saveFcmToken.spec.ts @@ -1,11 +1,11 @@ import "dotenv/config"; import type mongoose from "mongoose"; -import { User } from "../../../src/models"; +import { AppUserProfile } from "../../../src/models"; import type { MutationSaveFcmTokenArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import { saveFcmToken as saveFcmTokenResolver } from "../../../src/resolvers/Mutation/saveFcmToken"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; import type { TestUserType } from "../../helpers/user"; import { createTestUserFunc } from "../../helpers/user"; @@ -35,8 +35,8 @@ describe("resolvers -> Mutation -> saveFcmToken", () => { expect(saveFcmTokenPayload).toEqual(true); - const testSaveFcmTokenPayload = await User.findOne({ - _id: testUser?._id, + const testSaveFcmTokenPayload = await AppUserProfile.findOne({ + userId: testUser?._id, }) .select("token") .lean(); diff --git a/tests/resolvers/Mutation/signUp.spec.ts b/tests/resolvers/Mutation/signUp.spec.ts index 68223ee345..3b56351c14 100644 --- a/tests/resolvers/Mutation/signUp.spec.ts +++ b/tests/resolvers/Mutation/signUp.spec.ts @@ -1,34 +1,35 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User } from "../../../src/models"; -import type { MutationSignUpArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; -import { - LAST_RESORT_SUPERADMIN_EMAIL, - ORGANIZATION_NOT_FOUND_ERROR, -} from "../../../src/constants"; import { nanoid } from "nanoid"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, + expect, it, vi, - expect, - afterEach, } from "vitest"; +import { + LAST_RESORT_SUPERADMIN_EMAIL, + ORGANIZATION_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile, User } from "../../../src/models"; +import { signUp as signUpResolverImage } from "../../../src/resolvers/Mutation/signUp"; +import type { MutationSignUpArgs } from "../../../src/types/generatedGraphQLTypes"; +import * as uploadEncodedImage from "../../../src/utilities/encodedImageStorage/uploadEncodedImage"; +import { connect, disconnect } from "../../helpers/db"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; import { createTestUserAndOrganization } from "../../helpers/userAndOrg"; -import * as uploadEncodedImage from "../../../src/utilities/encodedImageStorage/uploadEncodedImage"; -import { signUp as signUpResolverImage } from "../../../src/resolvers/Mutation/signUp"; const testImagePath = `${nanoid().toLowerCase()}test.png`; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; + let testOrganization: TestOrganizationType; vi.mock("../../utilities/uploadEncodedImage", () => ({ @@ -47,6 +48,7 @@ beforeAll(async () => { MONGOOSE_INSTANCE = await connect(); const temp = await createTestUserAndOrganization(); testUser = temp[0]; + testOrganization = temp[1]; }); @@ -84,6 +86,7 @@ describe("resolvers -> Mutation -> signUp", () => { }) .select("-password") .lean(); + // console.log(createdUser, signUpPayload?.user); expect({ user: signUpPayload?.user, @@ -186,7 +189,10 @@ describe("resolvers -> Mutation -> signUp", () => { const createdUser = await User.findOne({ email, }); - expect(createdUser?.userType).toEqual("SUPERADMIN"); + const createdAppUserProfile = await AppUserProfile.findOne({ + userId: createdUser?._id, + }); + expect(createdAppUserProfile?.isSuperAdmin).toEqual(true); expect(createdUser?.adminApproved).toBeTruthy(); }); it(`Check if the User is not being promoted to SUPER ADMIN automatically`, async () => { @@ -242,9 +248,9 @@ describe("resolvers -> Mutation -> signUp", () => { ); await signUpResolver?.({}, args, {}); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(EMAIL_MESSAGE); - expect(error.message).toEqual(EMAIL_MESSAGE); + expect((error as Error).message).toEqual(EMAIL_MESSAGE); } }); it(`throws NotFoundError message if no organization exists with _id === args.data.organizationUserBelongsToId`, async () => { @@ -271,9 +277,11 @@ describe("resolvers -> Mutation -> signUp", () => { ); await signUpResolver?.({}, args, {}); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); }); diff --git a/tests/resolvers/Mutation/togglePostPin.spec.ts b/tests/resolvers/Mutation/togglePostPin.spec.ts index 7b42435c29..f3d7e05c3f 100644 --- a/tests/resolvers/Mutation/togglePostPin.spec.ts +++ b/tests/resolvers/Mutation/togglePostPin.spec.ts @@ -1,25 +1,25 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { Organization, User, Post } from "../../../src/models"; +import { AppUserProfile, Organization, Post, User } from "../../../src/models"; import type { MutationTogglePostPinArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; import { - POST_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_TO_PIN, - LENGTH_VALIDATION_ERROR, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; +import { + LENGTH_VALIDATION_ERROR, + POST_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_TO_PIN, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; import type { TestPostType } from "../../helpers/posts"; import { createTestPost } from "../../helpers/posts"; import type { TestUserType } from "../../helpers/userAndOrg"; @@ -72,8 +72,8 @@ describe("resolvers -> Mutation -> togglePostPin", () => { ); await togglePostPinResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -99,8 +99,8 @@ describe("resolvers -> Mutation -> togglePostPin", () => { ); await togglePostPinResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${POST_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -125,9 +125,9 @@ describe("resolvers -> Mutation -> togglePostPin", () => { ); await togglePostPinResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_TO_PIN.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_TO_PIN.MESSAGE}` ); } @@ -221,8 +221,10 @@ describe("resolvers -> Mutation -> togglePostPin", () => { ); await togglePostPinResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(`Please provide a title to pin post`); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `Please provide a title to pin post` + ); } }); @@ -247,10 +249,66 @@ describe("resolvers -> Mutation -> togglePostPin", () => { ); await togglePostPinResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 256 characters in title` ); } }); + it("throws an error if user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + user: testUser?._id, + }); + const { requestContext } = await import("../../../src/libraries"); + + vi.spyOn(requestContext, "translate").mockImplementationOnce( + (message) => message + ); + try { + const args: MutationTogglePostPinArgs = { + id: testPost?._id, + title: "Test title", + }; + + const context = { + userId: testUser?._id, + }; + + const { togglePostPin: togglePostPinResolver } = await import( + "../../../src/resolvers/Mutation/togglePostPin" + ); + + await togglePostPinResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_TO_PIN.MESSAGE}` + ); + } + }); + it("throws an error if user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + user: testUser?._id, + }); + const { requestContext } = await import("../../../src/libraries"); + vi.spyOn(requestContext, "translate").mockImplementationOnce( + (message) => message + ); + const args: MutationTogglePostPinArgs = { + id: testPost?._id, + title: "Test title", + }; + const context = { + userId: testUser?._id, + }; + try { + const { togglePostPin: togglePostPinResolver } = await import( + "../../../src/resolvers/Mutation/togglePostPin" + ); + await togglePostPinResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `${USER_NOT_AUTHORIZED_TO_PIN.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Mutation/unassignUserTag.spec.ts b/tests/resolvers/Mutation/unassignUserTag.spec.ts index 6fee16804b..cbf723faea 100644 --- a/tests/resolvers/Mutation/unassignUserTag.spec.ts +++ b/tests/resolvers/Mutation/unassignUserTag.spec.ts @@ -5,25 +5,25 @@ import type { MutationUnassignUserTagArgs } from "../../../src/types/generatedGr import { connect, disconnect } from "../../helpers/db"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, - TAG_NOT_FOUND, - USER_DOES_NOT_HAVE_THE_TAG, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; -import { createTestUser } from "../../helpers/userAndOrg"; +import { + TAG_NOT_FOUND, + USER_DOES_NOT_HAVE_THE_TAG, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile, TagUser } from "../../../src/models"; import type { TestUserTagType } from "../../helpers/tags"; import { createRootTagWithOrg } from "../../helpers/tags"; -import { TagUser } from "../../../src/models"; +import type { TestUserType } from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -70,8 +70,8 @@ describe("resolvers -> Mutation -> unassignUserTag", () => { ); await unassignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -100,8 +100,8 @@ describe("resolvers -> Mutation -> unassignUserTag", () => { ); await unassignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -132,9 +132,11 @@ describe("resolvers -> Mutation -> unassignUserTag", () => { ); await unassignUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(TAG_NOT_FOUND.MESSAGE); - expect(error.message).toEqual(`Translated ${TAG_NOT_FOUND.MESSAGE}`); + expect((error as Error).message).toEqual( + `Translated ${TAG_NOT_FOUND.MESSAGE}` + ); } }); @@ -162,8 +164,8 @@ describe("resolvers -> Mutation -> unassignUserTag", () => { ); await unassignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -195,8 +197,8 @@ describe("resolvers -> Mutation -> unassignUserTag", () => { ); await unassignUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_DOES_NOT_HAVE_THE_TAG.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -242,4 +244,36 @@ describe("resolvers -> Mutation -> unassignUserTag", () => { expect(tagAssigned).toBeFalsy(); }); + it("throws an error if the user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + try { + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const args: MutationUnassignUserTagArgs = { + input: { + userId: adminUser?._id, + tagId: testTag ? testTag._id.toString() : "", + }, + }; + const context = { + userId: newUser?._id, + }; + + const { unassignUserTag: unassignUserTagResolver } = await import( + "../../../src/resolvers/Mutation/unassignUserTag" + ); + + await unassignUserTagResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toHaveBeenLastCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/unregisterForEventByUser.spec.ts b/tests/resolvers/Mutation/unregisterForEventByUser.spec.ts index 5d47dbe1a7..06ab2dd012 100644 --- a/tests/resolvers/Mutation/unregisterForEventByUser.spec.ts +++ b/tests/resolvers/Mutation/unregisterForEventByUser.spec.ts @@ -62,9 +62,9 @@ describe("resolvers -> Mutation -> unregisterForEventByUser", () => { ); await unregisterForEventByUserResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${EVENT_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -78,7 +78,7 @@ describe("resolvers -> Mutation -> unregisterForEventByUser", () => { try { const args: MutationUnregisterForEventByUserArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", }; const context = { @@ -91,9 +91,9 @@ describe("resolvers -> Mutation -> unregisterForEventByUser", () => { ); await unregisterForEventByUserResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(USER_ALREADY_UNREGISTERED_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_ALREADY_UNREGISTERED_ERROR.MESSAGE}` ); } @@ -102,8 +102,8 @@ describe("resolvers -> Mutation -> unregisterForEventByUser", () => { it(`unregisters current user with _id === context.userId from event with _id === args.id`, async () => { await EventAttendee.create({ - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id, }); await User.updateOne( @@ -118,7 +118,7 @@ describe("resolvers -> Mutation -> unregisterForEventByUser", () => { ); const args: MutationUnregisterForEventByUserArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", }; const context = { @@ -131,8 +131,8 @@ describe("resolvers -> Mutation -> unregisterForEventByUser", () => { await unregisterForEventByUserResolver?.({}, args, context); const isUserRegistered = await EventAttendee.exists({ - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id, }); expect(isUserRegistered).toBeFalsy(); diff --git a/tests/resolvers/Mutation/updateActionItem.spec.ts b/tests/resolvers/Mutation/updateActionItem.spec.ts index db74533b50..ccf5bccd9a 100644 --- a/tests/resolvers/Mutation/updateActionItem.spec.ts +++ b/tests/resolvers/Mutation/updateActionItem.spec.ts @@ -1,32 +1,32 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import type { MutationUpdateActionItemArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { - USER_NOT_FOUND_ERROR, ACTION_ITEM_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, EVENT_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, USER_NOT_MEMBER_FOR_ORGANIZATION, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { - createTestUser, - createTestUserAndOrganization, -} from "../../helpers/userAndOrg"; import { updateActionItem as updateActionItemResolver } from "../../../src/resolvers/Mutation/updateActionItem"; +import type { MutationUpdateActionItemArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; +import { + createTestUser, + createTestUserAndOrganization, +} from "../../helpers/userAndOrg"; -import type { TestActionItemCategoryType } from "../../helpers/actionItemCategory"; -import { ActionItem, Event, User } from "../../../src/models"; +import { nanoid } from "nanoid"; +import { ActionItem, AppUserProfile, Event, User } from "../../../src/models"; import type { TestActionItemType } from "../../helpers/actionItem"; import { createTestActionItem } from "../../helpers/actionItem"; +import type { TestActionItemCategoryType } from "../../helpers/actionItemCategory"; import type { TestEventType } from "../../helpers/events"; -import { nanoid } from "nanoid"; let randomUser: TestUserType; let assignedTestUser: TestUserType; @@ -84,8 +84,8 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }; await updateActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -103,8 +103,10 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }; await updateActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ACTION_ITEM_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ACTION_ITEM_NOT_FOUND_ERROR.MESSAGE + ); } }); @@ -122,8 +124,8 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }; await updateActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -141,8 +143,10 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }; await updateActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_MEMBER_FOR_ORGANIZATION.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_MEMBER_FOR_ORGANIZATION.MESSAGE + ); } }); @@ -160,8 +164,10 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }; await updateActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); @@ -172,7 +178,7 @@ describe("resolvers -> Mutation -> updateActionItem", () => { assigneeId: assignedTestUser?._id, }, }; - + // console.log(testUser?._id); const context = { userId: testUser?._id, }; @@ -192,12 +198,12 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }); it(`updates the action item and returns it as superadmin`, async () => { - const superAdminTestUser = await User.findOneAndUpdate( + const superAdminTestUser = await AppUserProfile.findOneAndUpdate( { - _id: randomUser?._id, + userId: randomUser?._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, }, { new: true, @@ -212,7 +218,7 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }; const context = { - userId: superAdminTestUser?._id, + userId: superAdminTestUser?.userId, }; const updatedActionItemPayload = await updateActionItemResolver?.( @@ -264,8 +270,8 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }; await updateActionItemResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(EVENT_NOT_FOUND_ERROR.MESSAGE); } }); @@ -306,4 +312,25 @@ describe("resolvers -> Mutation -> updateActionItem", () => { }) ); }); + it("throws error if user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testUser2?._id, + }); + const args: MutationUpdateActionItemArgs = { + id: testActionItem?._id, + data: { + assigneeId: testUser?._id, + }, + }; + const context = { + userId: testUser2?._id, + }; + try { + await updateActionItemResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/updateActionItemCategory.spec.ts b/tests/resolvers/Mutation/updateActionItemCategory.spec.ts index e2718e69cc..497bb1fce2 100644 --- a/tests/resolvers/Mutation/updateActionItemCategory.spec.ts +++ b/tests/resolvers/Mutation/updateActionItemCategory.spec.ts @@ -1,24 +1,24 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import type { MutationUpdateActionItemCategoryArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ADMIN, ACTION_ITEM_CATEGORY_NOT_FOUND_ERROR, + USER_NOT_AUTHORIZED_ADMIN, + USER_NOT_FOUND_ERROR, } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import { createTestUser } from "../../helpers/userAndOrg"; +import type { MutationUpdateActionItemCategoryArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import type { TestOrganizationType, TestUserType, } from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; +import { AppUserProfile } from "../../../src/models"; import { updateActionItemCategory as updateActionItemCategoryResolver } from "../../../src/resolvers/Mutation/updateActionItemCategory"; import type { TestActionItemCategoryType } from "../../helpers/actionItemCategory"; import { createTestCategory } from "../../helpers/actionItemCategory"; -import { User } from "../../../src/models"; let randomUser: TestUserType; let testUser: TestUserType; @@ -58,8 +58,8 @@ describe("resolvers -> Mutation -> updateActionItemCategoryResolver", () => { }; await updateActionItemCategoryResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); @@ -78,8 +78,8 @@ describe("resolvers -> Mutation -> updateActionItemCategoryResolver", () => { }; await updateActionItemCategoryResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( ACTION_ITEM_CATEGORY_NOT_FOUND_ERROR.MESSAGE ); } @@ -100,8 +100,10 @@ describe("resolvers -> Mutation -> updateActionItemCategoryResolver", () => { }; await updateActionItemCategoryResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); @@ -134,12 +136,12 @@ describe("resolvers -> Mutation -> updateActionItemCategoryResolver", () => { }); it(`updates the actionItemCategory and returns it as superadmin`, async () => { - const superAdminTestUser = await User.findOneAndUpdate( + const superAdminTestUser = await AppUserProfile.findOneAndUpdate( { - _id: randomUser?._id, + userId: randomUser?._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, }, { new: true, @@ -155,7 +157,7 @@ describe("resolvers -> Mutation -> updateActionItemCategoryResolver", () => { }; const context = { - userId: superAdminTestUser?._id, + userId: superAdminTestUser?.userId, }; const updatedCategory = await updateActionItemCategoryResolver?.( diff --git a/tests/resolvers/Mutation/updateEvent.spec.ts b/tests/resolvers/Mutation/updateEvent.spec.ts index 1575968a10..c2216b247f 100644 --- a/tests/resolvers/Mutation/updateEvent.spec.ts +++ b/tests/resolvers/Mutation/updateEvent.spec.ts @@ -1,29 +1,29 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { User, Event } from "../../../src/models"; +import { AppUserProfile, Event, User } from "../../../src/models"; import type { MutationUpdateEventArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; import { - EVENT_NOT_FOUND_ERROR, - LENGTH_VALIDATION_ERROR, - USER_NOT_AUTHORIZED_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; +import { + EVENT_NOT_FOUND_ERROR, + LENGTH_VALIDATION_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { cacheEvents } from "../../../src/services/EventCache/cacheEvents"; +import type { TestEventType } from "../../helpers/events"; import type { TestUserType } from "../../helpers/userAndOrg"; import { createTestUserAndOrganization } from "../../helpers/userAndOrg"; -import type { TestEventType } from "../../helpers/events"; -import { cacheEvents } from "../../../src/services/EventCache/cacheEvents"; let MONGOOSE_INSTANCE: typeof mongoose; let testUser: TestUserType; @@ -89,9 +89,9 @@ describe("resolvers -> Mutation -> updateEvent", () => { ); await updateEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -117,9 +117,9 @@ describe("resolvers -> Mutation -> updateEvent", () => { ); await updateEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${EVENT_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -134,7 +134,7 @@ describe("resolvers -> Mutation -> updateEvent", () => { try { const args: MutationUpdateEventArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", }; const context = { @@ -146,9 +146,9 @@ describe("resolvers -> Mutation -> updateEvent", () => { ); await updateEventResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); } @@ -173,7 +173,7 @@ describe("resolvers -> Mutation -> updateEvent", () => { await cacheEvents([updatedEvent]); } - await User.updateOne( + await AppUserProfile.updateOne( { _id: testUser?._id, }, @@ -185,7 +185,7 @@ describe("resolvers -> Mutation -> updateEvent", () => { ); const args: MutationUpdateEventArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", data: { allDay: false, description: "newDescription", @@ -230,7 +230,7 @@ describe("Check for validation conditions", () => { ); try { const args: MutationUpdateEventArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", data: { allDay: false, description: "Random", @@ -259,8 +259,8 @@ describe("Check for validation conditions", () => { ); await updateEventResolverError?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 256 characters in title` ); } @@ -272,7 +272,7 @@ describe("Check for validation conditions", () => { ); try { const args: MutationUpdateEventArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", data: { allDay: false, description: @@ -301,8 +301,8 @@ describe("Check for validation conditions", () => { ); await updateEventResolverError?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 500 characters in description` ); } @@ -314,7 +314,7 @@ describe("Check for validation conditions", () => { ); try { const args: MutationUpdateEventArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", data: { allDay: false, description: "Random", @@ -342,8 +342,8 @@ describe("Check for validation conditions", () => { ); await updateEventResolverError?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `${LENGTH_VALIDATION_ERROR.MESSAGE} 50 characters in location` ); } @@ -355,7 +355,7 @@ describe("Check for validation conditions", () => { ); try { const args: MutationUpdateEventArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", data: { allDay: false, description: "Random", @@ -383,8 +383,54 @@ describe("Check for validation conditions", () => { ); await updateEventResolverError?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(`start date must be earlier than end date`); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `start date must be earlier than end date` + ); + } + }); + it("throws as error if user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + const { requestContext } = await import("../../../src/libraries"); + vi.spyOn(requestContext, "translate").mockImplementation( + (message) => message + ); + try { + const args: MutationUpdateEventArgs = { + id: testEvent?._id.toString() ?? "", + data: { + allDay: false, + description: "Random", + endDate: "Tue Feb 15 2023", + endTime: "", + isPublic: false, + isRegisterable: false, + latitude: 1, + longitude: 1, + location: "Random", + recurring: false, + startDate: "Tue Feb 14 2023", + startTime: "", + title: "Random", + recurrance: "DAILY", + }, + }; + + const context = { + userId: testUser?.id, + }; + + const { updateEvent: updateEventResolverError } = await import( + "../../../src/resolvers/Mutation/updateEvent" + ); + + await updateEventResolverError?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); } }); }); diff --git a/tests/resolvers/Mutation/updateUserPassword.spec.ts b/tests/resolvers/Mutation/updateUserPassword.spec.ts index 4cf36b17d3..094cf9fae7 100644 --- a/tests/resolvers/Mutation/updateUserPassword.spec.ts +++ b/tests/resolvers/Mutation/updateUserPassword.spec.ts @@ -1,31 +1,31 @@ import "dotenv/config"; -import type { Document } from "mongoose"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import type { InterfaceUser } from "../../../src/models"; -import { User } from "../../../src/models"; +import { AppUserProfile, User } from "../../../src/models"; import type { MutationUpdateUserPasswordArgs } from "../../../src/types/generatedGraphQLTypes"; import { connect, disconnect } from "../../helpers/db"; -import { updateUserPassword as updateUserPasswordResolver } from "../../../src/resolvers/Mutation/updateUserPassword"; -import { - INVALID_CREDENTIALS_ERROR, - USER_NOT_FOUND_ERROR, -} from "../../../src/constants"; +import bcrypt from "bcryptjs"; import { nanoid } from "nanoid"; import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; -import bcrypt from "bcryptjs"; +import { + INVALID_CREDENTIALS_ERROR, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { updateUserPassword as updateUserPasswordResolver } from "../../../src/resolvers/Mutation/updateUserPassword"; +import { createTestUser, type TestUserType } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; -let testUser: InterfaceUser & Document; +let testUser: TestUserType; vi.mock("../../utilities/uploadEncodedImage", () => ({ uploadEncodedImage: vi.fn(), @@ -41,7 +41,9 @@ beforeAll(async () => { password: hashedPassword, firstName: "firstName", lastName: "lastName", - appLanguageCode: "en", + }); + await AppUserProfile.create({ + userId: testUser._id, }); }); @@ -81,9 +83,9 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { ); await updateUserPasswordResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); } @@ -106,15 +108,17 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { }; const context = { - userId: testUser._id, + userId: testUser?._id, }; const { updateUserPassword: updateUserPasswordResolver } = await import( "../../../src/resolvers/Mutation/updateUserPassword" ); await updateUserPasswordResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(INVALID_CREDENTIALS_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + INVALID_CREDENTIALS_ERROR.MESSAGE + ); } }); @@ -135,7 +139,7 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { }; const context = { - userId: testUser._id, + userId: testUser?._id, }; const { updateUserPassword: updateUserPasswordResolver } = await import( @@ -143,8 +147,10 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { ); await updateUserPasswordResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(INVALID_CREDENTIALS_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + INVALID_CREDENTIALS_ERROR.MESSAGE + ); } }); @@ -165,7 +171,7 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { }; const context = { - userId: testUser._id, + userId: testUser?._id, }; const { updateUserPassword: updateUserPasswordResolver } = await import( @@ -173,8 +179,10 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { ); await updateUserPasswordResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(INVALID_CREDENTIALS_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + INVALID_CREDENTIALS_ERROR.MESSAGE + ); } }); @@ -188,7 +196,7 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { }; const context = { - userId: testUser._id, + userId: testUser?._id, }; const updateUserPasswordPayload = await updateUserPasswordResolver?.( @@ -199,4 +207,30 @@ describe("resolvers -> Mutation -> updateUserPassword", () => { expect(updateUserPasswordPayload).not.toBeNull(); }); + it("throws error if user does not have appLanguageCode", async () => { + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const args: MutationUpdateUserPasswordArgs = { + data: { + previousPassword: "password", + newPassword: "abcdabcd", + confirmNewPassword: "abcdabcd", + }, + }; + const context = { + userId: newUser?._id, + }; + + try { + await updateUserPasswordResolver?.({}, args, context); + } catch (error: unknown) { + console.log((error as Error).message); + // expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/updateUserRoleInOrganization.spec.ts b/tests/resolvers/Mutation/updateUserRoleInOrganization.spec.ts index deef8e6e90..6cb33e0ab3 100644 --- a/tests/resolvers/Mutation/updateUserRoleInOrganization.spec.ts +++ b/tests/resolvers/Mutation/updateUserRoleInOrganization.spec.ts @@ -1,7 +1,8 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { Organization, User } from "../../../src/models"; +import type { InterfaceAppUserProfile } from "../../../src/models"; +import { AppUserProfile, Organization, User } from "../../../src/models"; import { connect, disconnect } from "../../helpers/db"; import bcrypt from "bcryptjs"; @@ -20,19 +21,28 @@ import { ADMIN_CHANGING_ROLE_OF_CREATOR, ORGANIZATION_NOT_FOUND_ERROR, USER_NOT_AUTHORIZED_ADMIN, + USER_NOT_AUTHORIZED_ERROR, USER_NOT_FOUND_ERROR, USER_NOT_MEMBER_FOR_ORGANIZATION, } from "../../../src/constants"; import type { MutationUpdateUserRoleInOrganizationArgs } from "../../../src/types/generatedGraphQLTypes"; import type { TestUserType } from "../../helpers/user"; -import type { TestOrganizationType } from "../../helpers/userAndOrg"; +import type { + TestAppUserProfileType, + TestOrganizationType, +} from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let testUserSuperAdmin: TestUserType; +let testUserSuperAdminAppProfile: TestAppUserProfileType; let testNonMemberAdmin: TestUserType; +let testNonMemberAdminAppProfile: TestAppUserProfileType; let testMemberUser: TestUserType; +let testMemberUserAppProfile: TestAppUserProfileType; let testAdminUser: TestUserType; +let testAdminUserAppProfile: TestAppUserProfileType; let testBlockedMemberUser: TestUserType; +let testBlockedMemberUserAppProfile: TestAppUserProfileType; let testOrganization: TestOrganizationType; let hashedPassword: string; @@ -44,47 +54,98 @@ beforeAll(async () => { password: hashedPassword, firstName: "firstName", lastName: "lastName", + adminApproved: true, + }); + testUserSuperAdminAppProfile = await AppUserProfile.create({ + userId: testUserSuperAdmin._id, appLanguageCode: "en", - userType: "SUPERADMIN", isSuperAdmin: true, - adminApproved: true, }); + await User.updateOne( + { + _id: testUserSuperAdmin._id, + }, + { + appUserProfileId: testUserSuperAdminAppProfile._id, + } + ); + testAdminUser = await User.create({ email: `email${nanoid().toLowerCase()}@gmail.com`, password: hashedPassword, firstName: "firstName", lastName: "lastName", - appLanguageCode: "en", - userType: "ADMIN", + // appLanguageCode: "en", + // userType: "ADMIN", adminApproved: true, }); + testAdminUserAppProfile = await AppUserProfile.create({ + userId: testAdminUser._id, + appLanguageCode: "en", + }); + await User.updateOne( + { + _id: testAdminUser._id, + }, + { + appUserProfileId: testUserSuperAdminAppProfile._id, + } + ); + testMemberUser = await User.create({ email: `email${nanoid().toLowerCase()}@gmail.com`, password: hashedPassword, firstName: "firstName", lastName: "lastName", - appLanguageCode: "en", - userType: "USER", adminApproved: true, }); + testMemberUserAppProfile = await AppUserProfile.create({ + userId: testMemberUser._id, + }); + await User.updateOne( + { _id: testMemberUser._id }, + { + appUserProfileId: testMemberUserAppProfile._id, + } + ); + testBlockedMemberUser = await User.create({ email: `email${nanoid().toLowerCase()}@gmail.com`, password: hashedPassword, firstName: "firstName", lastName: "lastName", - appLanguageCode: "en", - userType: "USER", adminApproved: true, }); + testBlockedMemberUserAppProfile = await AppUserProfile.create({ + userId: testBlockedMemberUser._id, + }); + await User.updateOne( + { _id: testBlockedMemberUser._id }, + { + appUserProfileId: testBlockedMemberUserAppProfile._id, + } + ); testNonMemberAdmin = await User.create({ email: `email${nanoid().toLowerCase()}@gmail.com`, password: hashedPassword, firstName: "firstName", lastName: "lastName", - appLanguageCode: "en", - userType: "ADMIN", + // appLanguageCode: "en", + // userType: "ADMIN", adminApproved: true, }); + testNonMemberAdminAppProfile = await AppUserProfile.create({ + userId: testNonMemberAdmin._id, + appLanguageCode: "en", + }); + await User.updateOne( + { + _id: testNonMemberAdmin._id, + }, + { + appUserProfileId: testNonMemberAdminAppProfile._id, + } + ); testOrganization = await Organization.create({ name: "name", description: "description", @@ -99,11 +160,20 @@ beforeAll(async () => { { _id: testUserSuperAdmin?._id, }, + { + $set: { + joinedOrganizations: [testOrganization?._id], + }, + } + ); + await AppUserProfile.updateOne( + { + _id: testUserSuperAdminAppProfile?._id, + }, { $set: { createdOrganizations: [testOrganization?._id], adminFor: [testOrganization?._id], - joinedOrganizations: [testOrganization?._id], }, } ); @@ -113,11 +183,20 @@ beforeAll(async () => { }, { $set: { - adminFor: [testOrganization?._id], joinedOrganizations: [testOrganization?._id], }, } ); + await AppUserProfile.updateOne( + { + _id: testAdminUserAppProfile?._id, + }, + { + $set: { + adminFor: [testOrganization?._id], + }, + } + ); await User.updateOne( { _id: testMemberUser?._id, @@ -147,11 +226,11 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: Types.ObjectId().toHexString(), - userId: testUserSuperAdmin?._id, + userId: testUserSuperAdmin?._id.toString() ?? "", role: "ADMIN", }; const context = { - userId: testUserSuperAdmin?._id, + userId: testUserSuperAdmin?._id || "", }; const { @@ -160,8 +239,10 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ORGANIZATION_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ORGANIZATION_NOT_FOUND_ERROR.MESSAGE + ); } }); it(`Check when user whose role to be changed does not exists`, async () => { @@ -185,8 +266,8 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); it(`Check when user whose role to be changed is not a member of the organization`, async () => { @@ -197,7 +278,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testNonMemberAdmin?._id, + userId: testNonMemberAdmin?._id.toString() ?? "", role: "USER", }; const context = { @@ -210,8 +291,10 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_MEMBER_FOR_ORGANIZATION.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_MEMBER_FOR_ORGANIZATION.MESSAGE + ); } }); it(`Check when logged in user does not exists`, async () => { @@ -222,7 +305,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testMemberUser?._id, + userId: testMemberUser?._id.toString() ?? "", role: "ADMIN", }; const context = { @@ -235,8 +318,8 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); } }); it(`Check when USER is trying to change role of an admin`, async () => { @@ -247,7 +330,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testAdminUser?._id, + userId: testAdminUser?._id.toString() ?? "", role: "USER", }; const context = { @@ -260,8 +343,10 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); it(`Check when ADMIN of another org is not allowed to change role`, async () => { @@ -272,7 +357,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testMemberUser?._id, + userId: testMemberUser?._id.toString() ?? "", role: "ADMIN", }; const context = { @@ -285,8 +370,10 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); it(`Check when logged in ADMIN member user is not allowed to change the user type to SUPERADMIN`, async () => { @@ -297,7 +384,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testMemberUser?._id, + userId: testMemberUser?._id.toString() ?? "", role: "SUPERADMIN", }; const context = { @@ -310,8 +397,10 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_AUTHORIZED_ADMIN.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ADMIN.MESSAGE + ); } }); it(`Check when logged in ADMIN member user is trying to change the role of the itself`, async () => { @@ -322,7 +411,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testAdminUser?._id, + userId: testAdminUser?._id.toString() ?? "", role: "USER", }; const context = { @@ -335,8 +424,10 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ADMIN_CANNOT_CHANGE_ITS_ROLE.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ADMIN_CANNOT_CHANGE_ITS_ROLE.MESSAGE + ); } }); it(`Check when logged in ADMIN member user is trying to change the role of the org creator`, async () => { @@ -347,7 +438,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { try { const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testUserSuperAdmin?._id, + userId: testUserSuperAdmin?._id.toString() ?? "", role: "USER", }; const context = { @@ -360,8 +451,10 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { "../../../src/resolvers/Mutation/updateUserRoleInOrganization" ); await updateUserRoleInOrganizationResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(ADMIN_CHANGING_ROLE_OF_CREATOR.MESSAGE); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + ADMIN_CHANGING_ROLE_OF_CREATOR.MESSAGE + ); } }); it(`Check when SUPERUSER is changing the role of a USER member to ADMIN`, async () => { @@ -371,7 +464,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { ); const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testMemberUser?._id, + userId: testMemberUser?._id.toString() ?? "", role: "ADMIN", }; const context = { @@ -387,15 +480,17 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { const updatedOrganization = await Organization.findOne({ _id: testOrganization?._id, }).lean(); - const updatedUser = await User.findOne({ - _id: testMemberUser?._id, + const updatedUser: InterfaceAppUserProfile = await AppUserProfile.findOne({ + userId: testMemberUser?._id, }).lean(); const updatedOrganizationCheck = updatedOrganization?.admins.some( (member) => member.equals(testMemberUser?._id) ); - const updatedUserCheck = updatedUser?.adminFor.some((organization) => - organization.equals(testOrganization?._id) + const updatedUserCheck: boolean = updatedUser?.adminFor.some( + (organization) => + organization && + Types.ObjectId(organization.toString()).equals(testOrganization?._id) ); expect(updatedOrganizationCheck).toBe(true); expect(updatedUserCheck).toBe(true); @@ -407,7 +502,7 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { ); const args: MutationUpdateUserRoleInOrganizationArgs = { organizationId: testOrganization?._id, - userId: testAdminUser?._id, + userId: testAdminUser?._id.toString() ?? "", role: "USER", }; const context = { @@ -423,18 +518,48 @@ describe("resolvers -> Mutation -> updateUserRoleInOrganization", () => { const updatedOrg = await Organization.findOne({ _id: testOrganization?._id, }).lean(); - const updatedUser = await User.findOne({ - _id: testAdminUser?._id, + const updatedUser = await AppUserProfile.findOne({ + userId: testAdminUser?._id, }).lean(); const updatedOrgCheck = updatedOrg?.admins.some((member) => member.equals(testAdminUser?._id) ); const updatedUserCheck = updatedUser?.adminFor.some((organization) => - organization.equals(testOrganization?._id) + Types.ObjectId(organization?.toString()).equals(testOrganization?._id) ); expect(updatedOrgCheck).toBe(false); expect(updatedUserCheck).toBe(false); }); + it("throws an error if the user does not have appUserProfile", async () => { + await AppUserProfile.deleteOne({ + userId: testMemberUser?._id, + }); + const { requestContext } = await import("../../../src/libraries"); + vi.spyOn(requestContext, "translate").mockImplementation( + (message) => message + ); + try { + const args: MutationUpdateUserRoleInOrganizationArgs = { + organizationId: testOrganization?._id, + userId: testMemberUser?._id.toString() ?? "", + role: "ADMIN", + }; + const context = { + userId: testMemberUser?._id, + }; + + const { + updateUserRoleInOrganization: updateUserRoleInOrganizationResolver, + } = await import( + "../../../src/resolvers/Mutation/updateUserRoleInOrganization" + ); + await updateUserRoleInOrganizationResolver?.({}, args, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } + }); }); diff --git a/tests/resolvers/Mutation/updateUserTag.spec.ts b/tests/resolvers/Mutation/updateUserTag.spec.ts index 9b6d946bcf..d00fdd0b7e 100644 --- a/tests/resolvers/Mutation/updateUserTag.spec.ts +++ b/tests/resolvers/Mutation/updateUserTag.spec.ts @@ -5,26 +5,26 @@ import type { MutationUpdateUserTagArgs } from "../../../src/types/generatedGrap import { connect, disconnect } from "../../helpers/db"; import { - USER_NOT_FOUND_ERROR, - USER_NOT_AUTHORIZED_ERROR, - TAG_NOT_FOUND, - NO_CHANGE_IN_TAG_NAME, - TAG_ALREADY_EXISTS, -} from "../../../src/constants"; -import { - beforeAll, afterAll, + afterEach, + beforeAll, describe, - it, expect, + it, vi, - afterEach, } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; -import { createTestUser } from "../../helpers/userAndOrg"; +import { + NO_CHANGE_IN_TAG_NAME, + TAG_ALREADY_EXISTS, + TAG_NOT_FOUND, + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile, OrganizationTagUser } from "../../../src/models"; import type { TestUserTagType } from "../../helpers/tags"; import { createRootTagsWithOrg } from "../../helpers/tags"; -import { OrganizationTagUser } from "../../../src/models"; +import type { TestUserType } from "../../helpers/userAndOrg"; +import { createTestUser } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -59,7 +59,7 @@ describe("resolvers -> Mutation -> updateUserTag", () => { try { const args: MutationUpdateUserTagArgs = { input: { - _id: testTag!._id.toString(), + _id: testTag?._id.toString() ?? "", name: "NewName", }, }; @@ -71,8 +71,8 @@ describe("resolvers -> Mutation -> updateUserTag", () => { ); await updateUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -95,7 +95,7 @@ describe("resolvers -> Mutation -> updateUserTag", () => { }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { updateUserTag: updateUserTagResolver } = await import( @@ -103,9 +103,11 @@ describe("resolvers -> Mutation -> updateUserTag", () => { ); await updateUserTagResolver?.({}, args, context); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toHaveBeenLastCalledWith(TAG_NOT_FOUND.MESSAGE); - expect(error.message).toEqual(`Translated ${TAG_NOT_FOUND.MESSAGE}`); + expect((error as Error).message).toEqual( + `Translated ${TAG_NOT_FOUND.MESSAGE}` + ); } }); @@ -119,13 +121,13 @@ describe("resolvers -> Mutation -> updateUserTag", () => { try { const args: MutationUpdateUserTagArgs = { input: { - _id: testTag!._id.toString(), + _id: testTag?._id.toString() ?? "", name: "NewName", }, }; const context = { - userId: randomUser!._id, + userId: randomUser?._id, }; const { updateUserTag: updateUserTagResolver } = await import( @@ -133,8 +135,8 @@ describe("resolvers -> Mutation -> updateUserTag", () => { ); await updateUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -153,13 +155,13 @@ describe("resolvers -> Mutation -> updateUserTag", () => { try { const args: MutationUpdateUserTagArgs = { input: { - _id: testTag!._id.toString(), - name: testTag!.name, + _id: testTag?._id.toString() ?? "", + name: testTag?.name ?? "", }, }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { updateUserTag: updateUserTagResolver } = await import( @@ -167,8 +169,8 @@ describe("resolvers -> Mutation -> updateUserTag", () => { ); await updateUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${NO_CHANGE_IN_TAG_NAME.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(`${NO_CHANGE_IN_TAG_NAME.MESSAGE}`); @@ -185,13 +187,13 @@ describe("resolvers -> Mutation -> updateUserTag", () => { try { const args: MutationUpdateUserTagArgs = { input: { - _id: testTag!._id.toString(), - name: testTag2!.name, + _id: testTag?._id.toString() ?? "", + name: testTag2?.name ?? "", }, }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { updateUserTag: updateUserTagResolver } = await import( @@ -199,8 +201,10 @@ describe("resolvers -> Mutation -> updateUserTag", () => { ); await updateUserTagResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual(`Translated ${TAG_ALREADY_EXISTS.MESSAGE}`); + } catch (error: unknown) { + expect((error as Error).message).toEqual( + `Translated ${TAG_ALREADY_EXISTS.MESSAGE}` + ); expect(spy).toHaveBeenLastCalledWith(`${TAG_ALREADY_EXISTS.MESSAGE}`); } }); @@ -208,12 +212,12 @@ describe("resolvers -> Mutation -> updateUserTag", () => { it(`updates the task with _id === args.id and returns it`, async () => { const args: MutationUpdateUserTagArgs = { input: { - _id: testTag!._id.toString(), + _id: testTag?._id.toString() ?? "", name: "NewName", }, }; const context = { - userId: testUser!._id, + userId: testUser?._id, }; const { updateUserTag: updateUserTagResolver } = await import( @@ -223,9 +227,54 @@ describe("resolvers -> Mutation -> updateUserTag", () => { await updateUserTagResolver?.({}, args, context); const updatedTag = await OrganizationTagUser.findOne({ - _id: testTag!._id, + _id: testTag?._id, }).lean(); - expect(updatedTag!.name).toEqual("NewName"); + expect(updatedTag?.name).toEqual("NewName"); + }); + it("throws error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + + const args: MutationUpdateUserTagArgs = { + input: { + _id: testTag?._id.toString() ?? "", + name: "NewName", + }, + }; + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const context = { + userId: newUser?._id, + }; + + const { updateUserTag: updateUserTagResolver } = await import( + "../../../src/resolvers/Mutation/updateUserTag" + ); + + await OrganizationTagUser.updateOne( + { + _id: testTag?._id, + }, + { + $set: { + appUserProfileId: null, + }, + } + ); + + try { + await updateUserTagResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${USER_NOT_AUTHORIZED_ERROR.MESSAGE}` + ); + } }); }); diff --git a/tests/resolvers/Mutation/updateUserType.spec.ts b/tests/resolvers/Mutation/updateUserType.spec.ts deleted file mode 100644 index 9de06f751d..0000000000 --- a/tests/resolvers/Mutation/updateUserType.spec.ts +++ /dev/null @@ -1,224 +0,0 @@ -import "dotenv/config"; -import type mongoose from "mongoose"; -import { Types } from "mongoose"; -import { User } from "../../../src/models"; -import type { MutationUpdateUserTypeArgs } from "../../../src/types/generatedGraphQLTypes"; -import { connect, disconnect } from "../../helpers/db"; - -import { - USER_NOT_AUTHORIZED_SUPERADMIN, - USER_NOT_FOUND_ERROR, - SUPERADMIN_CANT_CHANGE_OWN_ROLE, -} from "../../../src/constants"; -import { - beforeAll, - afterAll, - afterEach, - describe, - it, - vi, - expect, -} from "vitest"; -import type { TestUserType } from "../../helpers/user"; -import { createTestUserFunc } from "../../helpers/user"; - -let MONGOOSE_INSTANCE: typeof mongoose; -let testUsers: TestUserType[]; - -beforeAll(async () => { - MONGOOSE_INSTANCE = await connect(); - const user1 = await createTestUserFunc(); - const user2 = await createTestUserFunc(); - testUsers = [user1, user2]; -}); - -afterAll(async () => { - await disconnect(MONGOOSE_INSTANCE); -}); - -afterEach(() => { - vi.doUnmock("../../../src/constants"); - vi.resetModules(); -}); - -describe("resolvers -> Mutation -> updateUserType", () => { - it(`throws NotFoundError if no user exists with _id === context.userId`, async () => { - const { requestContext } = await import("../../../src/libraries"); - const spy = vi - .spyOn(requestContext, "translate") - .mockImplementation((message) => `Translated ${message}`); - - try { - const args: MutationUpdateUserTypeArgs = { - data: { - id: Types.ObjectId().toString(), - }, - }; - - const context = { - userId: Types.ObjectId().toString(), - }; - - const { updateUserType: updateUserTypeResolver } = await import( - "../../../src/resolvers/Mutation/updateUserType" - ); - - await updateUserTypeResolver?.({}, args, context); - } catch (error: any) { - expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( - `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` - ); - } - }); - - it(`throws USER not super admin error if no user with _id === context.userId is not a SUPERADMIN`, async () => { - const { requestContext } = await import("../../../src/libraries"); - const spy = vi - .spyOn(requestContext, "translate") - .mockImplementation((message) => `Translated ${message}`); - - try { - const args: MutationUpdateUserTypeArgs = { - data: { - id: Types.ObjectId().toString(), - }, - }; - - const context = { - userId: testUsers[0]?._id, - }; - - const { updateUserType: updateUserTypeResolver } = await import( - "../../../src/resolvers/Mutation/updateUserType" - ); - - await updateUserTypeResolver?.({}, args, context); - } catch (error: any) { - expect(spy).toHaveBeenCalledWith(USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE); - expect(error.message).toEqual( - `Translated ${USER_NOT_AUTHORIZED_SUPERADMIN.MESSAGE}` - ); - } - }); - - it(`throws NotFoundError if no user exists with _id === args.data.id`, async () => { - const { requestContext } = await import("../../../src/libraries"); - const spy = vi - .spyOn(requestContext, "translate") - .mockImplementation((message) => `Translated ${message}`); - - try { - await User.updateOne( - { - _id: testUsers[0]?._id, - }, - { - userType: "SUPERADMIN", - }, - { - new: true, - } - ); - - const args: MutationUpdateUserTypeArgs = { - data: { - id: Types.ObjectId().toString(), - }, - }; - - const context = { - userId: testUsers[0]?._id, - }; - - const { updateUserType: updateUserTypeResolver } = await import( - "../../../src/resolvers/Mutation/updateUserType" - ); - - await updateUserTypeResolver?.({}, args, context); - } catch (error: any) { - expect(spy).toHaveBeenCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); - expect(error.message).toEqual( - `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` - ); - } - }); - - it(`throws SUPERADMIN_CANT_CHANGE_OWN_ROLE_ERROR if the user is a superadmin and tries to downgrade their own role`, async () => { - const { requestContext } = await import("../../../src/libraries"); - const spy = vi - .spyOn(requestContext, "translate") - .mockImplementation((message) => `Translated ${message}`); - - try { - const args: MutationUpdateUserTypeArgs = { - data: { - id: testUsers[0]?._id.toString(), - }, - }; - - const context = { - userId: testUsers[0]?._id, - }; - - const { updateUserType: updateUserTypeResolver } = await import( - "../../../src/resolvers/Mutation/updateUserType" - ); - await updateUserTypeResolver?.({}, args, context); - } catch (error: any) { - expect(spy).toHaveBeenCalledWith(SUPERADMIN_CANT_CHANGE_OWN_ROLE.MESSAGE); - expect(error.message).toEqual( - `Translated ${SUPERADMIN_CANT_CHANGE_OWN_ROLE.MESSAGE}` - ); - } - }); - - it(`updates user.userType of user with _id === args.data.id to args.data.userType`, async () => { - const { requestContext } = await import("../../../src/libraries"); - vi.spyOn(requestContext, "translate").mockImplementation( - (message) => `Translated ${message}` - ); - - await User.updateOne( - { - _id: testUsers[0]?._id, - }, - { - userType: "SUPERADMIN", - }, - { - new: true, - } - ); - - const args: MutationUpdateUserTypeArgs = { - data: { - id: testUsers[1]?._id.toString(), - userType: "BLOCKED", - }, - }; - const context = { - userId: testUsers[0]?._id, - }; - - const { updateUserType: updateUserTypeResolver } = await import( - "../../../src/resolvers/Mutation/updateUserType" - ); - - const updateUserTypePayload = await updateUserTypeResolver?.( - {}, - args, - context - ); - - expect(updateUserTypePayload).toEqual(true); - - const updatedTestUser = await User.findOne({ - _id: testUsers[1]?._id, - }) - .select("userType") - .lean(); - - expect(updatedTestUser?.userType).toEqual("BLOCKED"); - }); -}); diff --git a/tests/resolvers/Query/actionItemsByEvent.spec.ts b/tests/resolvers/Query/actionItemsByEvent.spec.ts index 34f161eee7..0cf2ac4171 100644 --- a/tests/resolvers/Query/actionItemsByEvent.spec.ts +++ b/tests/resolvers/Query/actionItemsByEvent.spec.ts @@ -1,11 +1,11 @@ import "dotenv/config"; +import type mongoose from "mongoose"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import { ActionItem } from "../../../src/models"; -import { connect, disconnect } from "../../helpers/db"; -import type { QueryActionItemsByEventArgs } from "../../../src/types/generatedGraphQLTypes"; import { actionItemsByEvent as actionItemsByEventsResolver } from "../../../src/resolvers/Query/actionItemsByEvent"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; -import type mongoose from "mongoose"; +import type { QueryActionItemsByEventArgs } from "../../../src/types/generatedGraphQLTypes"; import { createTestActionItems } from "../../helpers/actionItem"; +import { connect, disconnect } from "../../helpers/db"; import type { TestEventType } from "../../helpers/events"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -23,7 +23,7 @@ afterAll(async () => { describe("resolvers -> Query -> actionItemsByEvent", () => { it(`returns list of all action items associated with an event`, async () => { const args: QueryActionItemsByEventArgs = { - eventId: testEvent?._id, + eventId: testEvent?._id.toString() ?? "", }; const actionItemsByEventPayload = await actionItemsByEventsResolver?.( diff --git a/tests/resolvers/Query/checkAuth.spec.ts b/tests/resolvers/Query/checkAuth.spec.ts index 1f44d09f28..f6ee499387 100644 --- a/tests/resolvers/Query/checkAuth.spec.ts +++ b/tests/resolvers/Query/checkAuth.spec.ts @@ -1,16 +1,16 @@ import "dotenv/config"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; import { Types } from "mongoose"; import { checkAuth as checkAuthResolver } from "../../../src/resolvers/Query/checkAuth"; +import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; let MONGOOSE_INSTANCE: typeof mongoose; +import { AppUserProfile, User } from "../../../src/models"; import { createTestUser } from "../../helpers/userAndOrg"; -import { User } from "../../../src/models"; beforeAll(async () => { MONGOOSE_INSTANCE = await connect(); }); @@ -27,8 +27,8 @@ describe("resolvers -> Query -> checkAuth", () => { }; await checkAuthResolver?.({}, {}, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.DESC); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.DESC); } }); @@ -72,4 +72,20 @@ describe("resolvers -> Query -> checkAuth", () => { image: `${context.apiRootUrl}${testUser?.image}`, }); }); + it("throws error if user does not have appUserProfile", async () => { + try { + const testUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: testUser?._id, + }); + + const context = { + userId: testUser?._id, + }; + await checkAuthResolver?.({}, {}, context); + } catch (error: unknown) { + console.log(error); + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.DESC); + } + }); }); diff --git a/tests/resolvers/Query/event.spec.ts b/tests/resolvers/Query/event.spec.ts index 53de4afb29..89287cfb06 100644 --- a/tests/resolvers/Query/event.spec.ts +++ b/tests/resolvers/Query/event.spec.ts @@ -1,25 +1,24 @@ import "dotenv/config"; -import { event as eventResolver } from "../../../src/resolvers/Query/event"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; import { Types } from "mongoose"; import { EVENT_NOT_FOUND_ERROR } from "../../../src/constants"; import { Event } from "../../../src/models"; +import { event as eventResolver } from "../../../src/resolvers/Query/event"; +import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import type { QueryEventArgs } from "../../../src/types/generatedGraphQLTypes"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; -import type { TestUserType } from "../../helpers/userAndOrg"; import type { TestEventType } from "../../helpers/events"; import { createTestEvent } from "../../helpers/events"; let MONGOOSE_INSTANCE: typeof mongoose; let testEvent: TestEventType; -let testUser: TestUserType; +// let testUser: TestUserType; beforeAll(async () => { MONGOOSE_INSTANCE = await connect(); const resultArray = await createTestEvent(); - testUser = resultArray[0]; + // testUser = resultArray[0]; testEvent = resultArray[2]; }); @@ -35,14 +34,14 @@ describe("resolvers -> Query -> event", () => { }; await eventResolver?.({}, args, {}); - } catch (error: any) { - expect(error.message).toEqual(EVENT_NOT_FOUND_ERROR.DESC); + } catch (error: unknown) { + expect((error as Error).message).toEqual(EVENT_NOT_FOUND_ERROR.DESC); } }); it(`returns event object with populated fields creator and admins`, async () => { const args: QueryEventArgs = { - id: testEvent?._id, + id: testEvent?._id.toString() ?? "", }; const eventPayload = await eventResolver?.({}, args, {}); diff --git a/tests/resolvers/Query/hasSubmittedFeedback.spec.ts b/tests/resolvers/Query/hasSubmittedFeedback.spec.ts index 10e903074e..e2e75b27e8 100644 --- a/tests/resolvers/Query/hasSubmittedFeedback.spec.ts +++ b/tests/resolvers/Query/hasSubmittedFeedback.spec.ts @@ -1,18 +1,18 @@ import "dotenv/config"; -import { connect, disconnect } from "../../helpers/db"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import type { QueryHasSubmittedFeedbackArgs } from "../../../src/types/generatedGraphQLTypes"; import type mongoose from "mongoose"; import { Types } from "mongoose"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; +import { CheckIn, EventAttendee } from "../../../src/models"; +import type { QueryHasSubmittedFeedbackArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; +import { createTestEvent, type TestEventType } from "../../helpers/events"; import { EVENT_NOT_FOUND_ERROR, - USER_NOT_FOUND_ERROR, USER_NOT_CHECKED_IN, + USER_NOT_FOUND_ERROR, USER_NOT_REGISTERED_FOR_EVENT, } from "./../../../src/constants"; -import { type TestUserType, createTestUser } from "./../../helpers/userAndOrg"; -import { createTestEvent, type TestEventType } from "../../helpers/events"; -import { CheckIn, EventAttendee } from "../../../src/models"; +import { createTestUser, type TestUserType } from "./../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; let randomTestUser: TestUserType; @@ -50,8 +50,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { await import("../../../src/resolvers/Query/hasSubmittedFeedback"); await hasSubmittedFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_FOUND_ERROR.MESSAGE); @@ -67,7 +67,7 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { try { const args: QueryHasSubmittedFeedbackArgs = { - userId: randomTestUser!._id, + userId: randomTestUser?._id, eventId: Types.ObjectId().toString(), }; @@ -77,8 +77,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { await import("../../../src/resolvers/Query/hasSubmittedFeedback"); await hasSubmittedFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${EVENT_NOT_FOUND_ERROR.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(EVENT_NOT_FOUND_ERROR.MESSAGE); @@ -94,8 +94,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { try { const args: QueryHasSubmittedFeedbackArgs = { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }; const context = {}; @@ -104,8 +104,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { await import("../../../src/resolvers/Query/hasSubmittedFeedback"); await hasSubmittedFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_REGISTERED_FOR_EVENT.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith( @@ -116,8 +116,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { it(`throws Error if the user is has not checked in to the event`, async () => { const eventAttendee = await EventAttendee.create({ - eventId: testEvent!._id, - userId: testUser!._id, + eventId: testEvent?._id, + userId: testUser?._id, }); eventAttendeeId = eventAttendee._id; @@ -129,8 +129,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { try { const args: QueryHasSubmittedFeedbackArgs = { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }; const context = {}; @@ -139,8 +139,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { await import("../../../src/resolvers/Query/hasSubmittedFeedback"); await hasSubmittedFeedbackResolver?.({}, args, context); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_CHECKED_IN.MESSAGE}` ); expect(spy).toHaveBeenLastCalledWith(USER_NOT_CHECKED_IN.MESSAGE); @@ -158,8 +158,8 @@ describe("resolvers -> Query -> hasSubmittedFeedback", () => { }); const args: QueryHasSubmittedFeedbackArgs = { - userId: testUser!._id, - eventId: testEvent!._id, + userId: testUser?._id, + eventId: testEvent?._id.toString() ?? "", }; const context = {}; diff --git a/tests/resolvers/Query/myLanguage.spec.ts b/tests/resolvers/Query/myLanguage.spec.ts index a0d608a8d2..19a4f281fe 100644 --- a/tests/resolvers/Query/myLanguage.spec.ts +++ b/tests/resolvers/Query/myLanguage.spec.ts @@ -1,13 +1,14 @@ import "dotenv/config"; -import { myLanguage as myLanguageResolver } from "../../../src/resolvers/Query/myLanguage"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; -import { User } from "../../../src/models"; import { nanoid } from "nanoid"; +import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; +import { AppUserProfile, User } from "../../../src/models"; +import { myLanguage as myLanguageResolver } from "../../../src/resolvers/Query/myLanguage"; +import { connect, disconnect } from "../../helpers/db"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; +import { createTestUser } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -27,8 +28,8 @@ describe("resolvers -> Query -> myLanguage", () => { }; await myLanguageResolver?.({}, {}, context); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.DESC); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.DESC); } }); @@ -38,8 +39,19 @@ describe("resolvers -> Query -> myLanguage", () => { password: "password", firstName: "firstName", lastName: "lastName", + }); + const testAppUserProfile = await AppUserProfile.create({ + userId: testUser._id, appLanguageCode: "en", }); + await User.updateOne( + { + _id: testUser._id, + }, + { + appUserProfileId: testAppUserProfile._id, + } + ); const context = { userId: testUser?._id, @@ -47,6 +59,21 @@ describe("resolvers -> Query -> myLanguage", () => { const appLanguageCodePayload = await myLanguageResolver?.({}, {}, context); - expect(appLanguageCodePayload).toEqual(testUser?.appLanguageCode); + expect(appLanguageCodePayload).toEqual(testAppUserProfile?.appLanguageCode); + }); + it("throws error if user does not have appLanguageCode", async () => { + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const context = { + userId: newUser?._id, + }; + + try { + await myLanguageResolver?.({}, {}, context); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.MESSAGE); + } }); }); diff --git a/tests/resolvers/Query/organizationsMemberConnection.spec.ts b/tests/resolvers/Query/organizationsMemberConnection.spec.ts index 75aed8a275..4050261a0c 100644 --- a/tests/resolvers/Query/organizationsMemberConnection.spec.ts +++ b/tests/resolvers/Query/organizationsMemberConnection.spec.ts @@ -1,21 +1,21 @@ import "dotenv/config"; -import { organizationsMemberConnection as organizationsMemberConnectionResolver } from "../../../src/resolvers/Query/organizationsMemberConnection"; -import type { InterfaceOrganization, InterfaceUser } from "../../../src/models"; -import { Organization, User } from "../../../src/models"; -import { connect, disconnect } from "../../helpers/db"; -import type { Document } from "mongoose"; import type mongoose from "mongoose"; +import type { Document } from "mongoose"; import { Types } from "mongoose"; +import type { InterfaceOrganization, InterfaceUser } from "../../../src/models"; +import { AppUserProfile, Organization, User } from "../../../src/models"; +import { organizationsMemberConnection as organizationsMemberConnectionResolver } from "../../../src/resolvers/Query/organizationsMemberConnection"; import type { QueryOrganizationsMemberConnectionArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import { nanoid } from "nanoid"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import { BASE_URL } from "../../../src/constants"; let MONGOOSE_INSTANCE: typeof mongoose; -let testUsers: (InterfaceUser & Document)[]; +let testUsers: (InterfaceUser & Document)[]; let testOrganization: InterfaceOrganization & - Document; + Document; beforeAll(async () => { MONGOOSE_INSTANCE = await connect(); @@ -26,23 +26,25 @@ beforeAll(async () => { password: "password", firstName: `1firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid().toLowerCase()}`, }, { email: `email${nanoid().toLowerCase()}@gmail.com`, password: "password", firstName: `2firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid().toLowerCase()}`, }, { email: `email${nanoid().toLowerCase()}@gmail.com`, password: "password", firstName: `3firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid().toLowerCase()}`, }, ]); + const appUserProfiles = testUsers.map((user) => ({ + userId: user._id, + appLanguageCode: `en${nanoid().toLowerCase()}`, + })); + await AppUserProfile.insertMany(appUserProfiles); testOrganization = await Organization.create({ name: "name", @@ -59,11 +61,20 @@ beforeAll(async () => { { _id: testUsers[0]._id, }, + { + $push: { + joinedOrganizations: testOrganization._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: testUsers[0]._id, + }, { $push: { createdOrganizations: testOrganization._id, adminFor: testOrganization._id, - joinedOrganizations: testOrganization._id, }, } ); @@ -74,11 +85,20 @@ beforeAll(async () => { }, { $push: { - adminFor: testOrganization._id, joinedOrganizations: testOrganization._id, }, } ); + await AppUserProfile.updateOne( + { + userid: testUsers[1]._id, + }, + { + $push: { + adminFor: testOrganization._id, + }, + } + ); await User.updateOne( { @@ -86,11 +106,20 @@ beforeAll(async () => { }, { $push: { - adminFor: [testOrganization._id], joinedOrganizations: [testOrganization._id], }, } ); + await AppUserProfile.updateOne( + { + userId: testUsers[2]._id, + }, + { + $push: { + adminFor: testOrganization._id, + }, + } + ); }); afterAll(async () => { @@ -136,7 +165,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }; const sort = { @@ -152,7 +181,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }, orderBy: "id_ASC", }; @@ -209,9 +238,9 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { email: { $ne: testUsers[2].email, }, - appLanguageCode: { - $ne: testUsers[2].appLanguageCode, - }, + // appLanguageCode: { + // $ne: testUsers[2].appLanguageCode, + // }, }; const sort = { @@ -223,11 +252,11 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { first: 2, skip: 1, where: { - id_not: testUsers[2]._id, + id_not: testUsers[2]._id.toString(), firstName_not: testUsers[2].firstName, lastName_not: testUsers[2].lastName, email_not: testUsers[2].email, - appLanguageCode_not: testUsers[2].appLanguageCode, + // appLanguageCode_not: testUsers[2].appLanguageCode, }, orderBy: "id_DESC", }; @@ -250,8 +279,8 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { id: String(user._id), }; }); - - expect(organizationsMemberConnectionPayload).toEqual({ + // console.log(organizationsMemberConnectionPayload, usersWithPassword); + expect(organizationsMemberConnectionPayload).toMatchObject({ pageInfo: { hasNextPage: false, hasPreviousPage: false, @@ -285,9 +314,9 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { email: { $in: [testUsers[1].email], }, - appLanguageCode: { - $in: [testUsers[1].appLanguageCode], - }, + // appLanguageCode: { + // $in: [testUsers[1].appLanguageCode], + // }, }; const sort = { @@ -295,7 +324,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: 1, where: { @@ -303,7 +332,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { firstName_in: [testUsers[1].firstName], lastName_in: [testUsers[1].lastName], email_in: [testUsers[1].email], - appLanguageCode_in: [testUsers[1].appLanguageCode], + // appLanguageCode_in: [testUsers[1].appLanguageCode], }, orderBy: "firstName_ASC", }; @@ -360,9 +389,9 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { email: { $nin: [testUsers[2].email], }, - appLanguageCode: { - $nin: [testUsers[2].appLanguageCode], - }, + // appLanguageCode: { + // $nin: [testUsers[2].appLanguageCode], + // }, joinedOrganizations: { $in: testOrganization._id, }, @@ -373,15 +402,15 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: 1, where: { - id_not_in: [testUsers[2]._id], + id_not_in: [testUsers[2]._id.toString()], firstName_not_in: [testUsers[2].firstName], lastName_not_in: [testUsers[2].lastName], email_not_in: [testUsers[2].email], - appLanguageCode_not_in: [testUsers[2].appLanguageCode], + // appLanguageCode_not_in: [testUsers[2].appLanguageCode], }, orderBy: "firstName_DESC", }; @@ -438,10 +467,10 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { $regex: testUsers[1].email, $options: "i", }, - appLanguageCode: { - $regex: testUsers[1].appLanguageCode, - $options: "i", - }, + // appLanguageCode: { + // $regex: testUsers[1].appLanguageCode, + // $options: "i", + // }, joinedOrganizations: { $in: testOrganization._id, }, @@ -452,14 +481,14 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: 1, where: { firstName_contains: testUsers[1].firstName, lastName_contains: testUsers[1].lastName, email_contains: testUsers[1].email, - appLanguageCode_contains: testUsers[1].appLanguageCode, + // appLanguageCode_contains: testUsers[1].appLanguageCode, }, orderBy: "lastName_ASC", }; @@ -507,7 +536,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { firstName: new RegExp("^" + testUsers[0].firstName), lastName: new RegExp("^" + testUsers[0].lastName), email: new RegExp("^" + testUsers[0].email), - appLanguageCode: new RegExp("^" + testUsers[0].appLanguageCode), + // appLanguageCode: new RegExp("^" + testUsers[0].appLanguageCode), joinedOrganizations: { $in: testOrganization._id, }, @@ -518,14 +547,14 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: 1, where: { firstName_starts_with: testUsers[0].firstName, lastName_starts_with: testUsers[0].lastName, email_starts_with: testUsers[0].email, - appLanguageCode_starts_with: testUsers[0].appLanguageCode, + // appLanguageCode_starts_with: testUsers[0].appLanguageCode, }, orderBy: "lastName_DESC", }; @@ -564,115 +593,115 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }); }); - it(`returns paginated list of users sorted by - args.orderBy === 'appLanguageCode_ASC'`, async () => { - const where = { - joinedOrganizations: { - $in: testOrganization._id, - }, - }; - - const sort = { - appLanguageCode: 1, - }; - - const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, - first: 2, - skip: 1, - where: {}, - orderBy: "appLanguageCode_ASC", - }; - - const organizationsMemberConnectionPayload = - await organizationsMemberConnectionResolver?.({}, args, {}); - - const users = await User.find(where) - .sort(sort) - .limit(2) - .select(["-password"]) - .populate(["registeredEvents"]) - .lean(); - - const usersWithPassword = users.map((user) => { - return { - ...user, - password: null, - image: user.image || null, - id: String(user._id), - }; - }); - - expect(organizationsMemberConnectionPayload).toEqual({ - pageInfo: { - hasNextPage: true, - hasPreviousPage: false, - totalPages: 2, - nextPageNo: 2, - prevPageNo: null, - currPageNo: 1, - }, - edges: usersWithPassword, - aggregate: { - count: 3, - }, - }); - }); - - it(`returns paginated list of users sorted by - args.orderBy === 'appLanguageCode_DESC'`, async () => { - const where = { - joinedOrganizations: { - $in: testOrganization._id, - }, - }; - - const sort = { - appLanguageCode: -1, - }; - - const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, - first: 2, - skip: 1, - where: {}, - orderBy: "appLanguageCode_DESC", - }; - - const organizationsMemberConnectionPayload = - await organizationsMemberConnectionResolver?.({}, args, {}); - - const users = await User.find(where) - .sort(sort) - .limit(2) - .select(["-password"]) - .populate(["registeredEvents"]) - .lean(); - - const usersWithPassword = users.map((user) => { - return { - ...user, - password: null, - image: user.image || null, - id: String(user._id), - }; - }); - - expect(organizationsMemberConnectionPayload).toEqual({ - pageInfo: { - hasNextPage: true, - hasPreviousPage: false, - totalPages: 2, - nextPageNo: 2, - prevPageNo: null, - currPageNo: 1, - }, - edges: usersWithPassword, - aggregate: { - count: 3, - }, - }); - }); + // it(`returns paginated list of users sorted by + // args.orderBy === 'appLanguageCode_ASC'`, async () => { + // const where = { + // joinedOrganizations: { + // $in: testOrganization._id, + // }, + // }; + + // const sort = { + // appLanguageCode: 1, + // }; + + // const args: QueryOrganizationsMemberConnectionArgs = { + // orgId: testOrganization._id.toString(), + // first: 2, + // skip: 1, + // where: {}, + // orderBy: "appLanguageCode_ASC", + // }; + + // const organizationsMemberConnectionPayload = + // await organizationsMemberConnectionResolver?.({}, args, {}); + + // const users = await User.find(where) + // .sort(sort) + // .limit(2) + // .select(["-password"]) + // .populate(["registeredEvents"]) + // .lean(); + + // const usersWithPassword = users.map((user) => { + // return { + // ...user, + // password: null, + // image: user.image || null, + // id: String(user._id), + // }; + // }); + + // expect(organizationsMemberConnectionPayload).toEqual({ + // pageInfo: { + // hasNextPage: true, + // hasPreviousPage: false, + // totalPages: 2, + // nextPageNo: 2, + // prevPageNo: null, + // currPageNo: 1, + // }, + // edges: usersWithPassword, + // aggregate: { + // count: 3, + // }, + // }); + // }); + + // it(`returns paginated list of users sorted by + // args.orderBy === 'appLanguageCode_DESC'`, async () => { + // const where = { + // joinedOrganizations: { + // $in: testOrganization._id, + // }, + // }; + + // const sort = { + // appLanguageCode: -1, + // }; + + // const args: QueryOrganizationsMemberConnectionArgs = { + // orgId: testOrganization._id, + // first: 2, + // skip: 1, + // where: {}, + // orderBy: "appLanguageCode_DESC", + // }; + + // const organizationsMemberConnectionPayload = + // await organizationsMemberConnectionResolver?.({}, args, {}); + + // const users = await User.find(where) + // .sort(sort) + // .limit(2) + // .select(["-password"]) + // .populate(["registeredEvents"]) + // .lean(); + + // const usersWithPassword = users.map((user) => { + // return { + // ...user, + // password: null, + // image: user.image || null, + // id: String(user._id), + // }; + // }); + + // expect(organizationsMemberConnectionPayload).toEqual({ + // pageInfo: { + // hasNextPage: true, + // hasPreviousPage: false, + // totalPages: 2, + // nextPageNo: 2, + // prevPageNo: null, + // currPageNo: 1, + // }, + // edges: usersWithPassword, + // aggregate: { + // count: 3, + // }, + // }); + // }); it(`returns paginated list of users sorted by args.orderBy === 'email_ASC'`, async () => { @@ -687,7 +716,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: 1, where: null, @@ -713,7 +742,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; }); - expect(organizationsMemberConnectionPayload).toEqual({ + expect(organizationsMemberConnectionPayload).toMatchObject({ pageInfo: { hasNextPage: true, hasPreviousPage: false, @@ -742,7 +771,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: 1, where: null, @@ -786,7 +815,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { it(`throws Error if args.skip === null`, async () => { const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: null, where: null, @@ -795,7 +824,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { try { await organizationsMemberConnectionResolver?.({}, args, {}); - } catch (error: any) { + } catch (error: unknown) { expect(error).toEqual( "Missing Skip parameter. Set it to either 0 or some other value" ); @@ -804,7 +833,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { it(`throws Error if args.skip === undefined`, async () => { const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: undefined, where: null, @@ -813,8 +842,8 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { try { await organizationsMemberConnectionResolver?.({}, args, {}); - } catch (error: any) { - expect(error.message).toEqual("Skip parameter is missing"); + } catch (error: unknown) { + expect((error as Error).message).toEqual("Skip parameter is missing"); } }); @@ -826,7 +855,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), skip: 1, where: {}, orderBy: null, @@ -886,7 +915,7 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }; const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), first: 2, skip: 1, where: null, @@ -930,71 +959,71 @@ describe("resolvers -> Query -> organizationsMemberConnection", () => { }, }); }); - it(`returns non-paginated list of admins if args.first === undefined and where.admin_for !== undefined`, async () => { - await User.updateMany( - {}, - { - $set: { - image: `image/image.png`, - }, - } - ); - const where = { - joinedOrganizations: { - $in: testOrganization._id, - }, - }; - - const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, - skip: 1, - where: { - admin_for: testOrganization._id, - }, - orderBy: null, - }; - - const context = { - apiRootUrl: BASE_URL, - }; - - const organizationsMemberConnectionPayload = - await organizationsMemberConnectionResolver?.({}, args, context); - - const usersTestModel = await User.paginate(where, { - pagination: false, - sort: {}, - populate: ["registeredEvents"], - select: ["-password"], - }); - - const users = usersTestModel.docs.map((user) => { - return { - ...user._doc, - image: `${BASE_URL}${user.image}`, - password: null, - }; - }); - - expect(organizationsMemberConnectionPayload).toEqual({ - pageInfo: { - hasNextPage: false, - hasPreviousPage: false, - totalPages: 1, - nextPageNo: null, - prevPageNo: null, - currPageNo: 1, - }, - edges: users, - aggregate: { - count: 3, - }, - }); - }); + // it(`returns non-paginated list of admins if args.first === undefined and where.admin_for !== undefined`, async () => { + // await User.updateMany( + // {}, + // { + // $set: { + // image: `image/image.png`, + // }, + // } + // ); + // const where = { + // joinedOrganizations: { + // $in: testOrganization._id, + // }, + // }; + + // const args: QueryOrganizationsMemberConnectionArgs = { + // orgId: testOrganization._id.toString(), + // skip: 1, + // where: { + // admin_for: testOrganization._id, + // }, + // orderBy: null, + // }; + + // const context = { + // apiRootUrl: BASE_URL, + // }; + + // const organizationsMemberConnectionPayload = + // await organizationsMemberConnectionResolver?.({}, args, context); + + // const usersTestModel = await User.paginate(where, { + // pagination: false, + // sort: {}, + // populate: ["registeredEvents"], + // select: ["-password"], + // }); + + // const users = usersTestModel.docs.map((user) => { + // return { + // ...user._doc, + // image: `${BASE_URL}${user.image}`, + // password: null, + // }; + // }); + + // expect(organizationsMemberConnectionPayload).toEqual({ + // pageInfo: { + // hasNextPage: false, + // hasPreviousPage: false, + // totalPages: 1, + // nextPageNo: null, + // prevPageNo: null, + // currPageNo: 1, + // }, + // edges: users, + // aggregate: { + // count: 3, + // }, + // }); + // }); it(`returns non-paginated list of admins if args.first === undefined and where.event_title_contains !== undefined`, async () => { const args: QueryOrganizationsMemberConnectionArgs = { - orgId: testOrganization._id, + orgId: testOrganization._id.toString(), skip: 1, where: { event_title_contains: "testEvent", diff --git a/tests/resolvers/Query/user.spec.ts b/tests/resolvers/Query/user.spec.ts index bade5dfd91..2e8d487c44 100644 --- a/tests/resolvers/Query/user.spec.ts +++ b/tests/resolvers/Query/user.spec.ts @@ -1,13 +1,13 @@ import "dotenv/config"; -import { user as userResolver } from "../../../src/resolvers/Query/user"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; import { Types } from "mongoose"; import { BASE_URL, USER_NOT_FOUND_ERROR } from "../../../src/constants"; import { User } from "../../../src/models"; +import { user as userResolver } from "../../../src/resolvers/Query/user"; +import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import type { QueryUserArgs } from "../../../src/types/generatedGraphQLTypes"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; import type { TestUserType } from "../../helpers/userAndOrg"; import { createTestUserAndOrganization } from "../../helpers/userAndOrg"; @@ -31,8 +31,8 @@ describe("resolvers -> Query -> user", () => { }; await userResolver?.({}, args, {}); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.DESC); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.DESC); } }); @@ -49,11 +49,9 @@ describe("resolvers -> Query -> user", () => { const user = await User.findOne({ _id: testUser?._id, - }) - .populate("adminFor") - .lean(); + }).lean(); - expect(userPayload).toEqual({ + expect(userPayload?.user).toEqual({ ...user, organizationsBlockedBy: [], image: null, @@ -84,11 +82,9 @@ describe("resolvers -> Query -> user", () => { const user = await User.findOne({ _id: testUser?._id, - }) - .populate("adminFor") - .lean(); + }).lean(); - expect(userPayload).toEqual({ + expect(userPayload?.user).toEqual({ ...user, organizationsBlockedBy: [], image: user?.image ? `${BASE_URL}${user.image}` : null, diff --git a/tests/resolvers/Query/userLanguage.spec.ts b/tests/resolvers/Query/userLanguage.spec.ts index 49b6c11e50..2e8614727c 100644 --- a/tests/resolvers/Query/userLanguage.spec.ts +++ b/tests/resolvers/Query/userLanguage.spec.ts @@ -1,14 +1,17 @@ import "dotenv/config"; import type mongoose from "mongoose"; import { Types } from "mongoose"; -import { nanoid } from "nanoid"; import { connect, disconnect } from "../../helpers/db"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; +import { + USER_NOT_AUTHORIZED_ERROR, + USER_NOT_FOUND_ERROR, +} from "../../../src/constants"; +import { AppUserProfile } from "../../../src/models"; import { userLanguage as userLanguageResolver } from "../../../src/resolvers/Query/userLanguage"; -import { USER_NOT_FOUND_ERROR } from "../../../src/constants"; -import { User } from "../../../src/models"; import type { QueryUserLanguageArgs } from "../../../src/types/generatedGraphQLTypes"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; +import { createTestUser } from "../../helpers/userAndOrg"; let MONGOOSE_INSTANCE: typeof mongoose; @@ -28,26 +31,45 @@ describe("resolvers -> Query -> userLanguage", () => { }; await userLanguageResolver?.({}, args, {}); - } catch (error: any) { - expect(error.message).toEqual(USER_NOT_FOUND_ERROR.DESC); + } catch (error: unknown) { + expect((error as Error).message).toEqual(USER_NOT_FOUND_ERROR.DESC); } }); it(`returns user's appLanguageCode`, async () => { - const testUser = await User.create({ - email: `email${nanoid().toLowerCase()}@gmail.com`, - password: "password", - firstName: "firstName", - lastName: "lastName", - appLanguageCode: "en", - }); + const testUser = await createTestUser(); const args: QueryUserLanguageArgs = { userId: testUser?._id, }; + const testAppUserProfile = await AppUserProfile.findOne({ + userId: testUser?._id, + }); const userLanguagePayload = await userLanguageResolver?.({}, args, {}); - expect(userLanguagePayload).toEqual(testUser.appLanguageCode); + expect(userLanguagePayload).toEqual(testAppUserProfile?.appLanguageCode); + }); + it("throws error if user does not have appLanguageCode", async () => { + const { requestContext } = await import("../../../src/libraries"); + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => message); + const newUser = await createTestUser(); + await AppUserProfile.deleteOne({ + userId: newUser?.id, + }); + const args: QueryUserLanguageArgs = { + userId: newUser?._id, + }; + + try { + await userLanguageResolver?.({}, args, {}); + } catch (error: unknown) { + expect(spy).toBeCalledWith(USER_NOT_AUTHORIZED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + USER_NOT_AUTHORIZED_ERROR.MESSAGE + ); + } }); }); diff --git a/tests/resolvers/Query/users.spec.ts b/tests/resolvers/Query/users.spec.ts index 9cc103f1ca..482774d4ba 100644 --- a/tests/resolvers/Query/users.spec.ts +++ b/tests/resolvers/Query/users.spec.ts @@ -1,17 +1,17 @@ import "dotenv/config"; -import { users as usersResolver } from "../../../src/resolvers/Query/users"; -import type { InterfaceUser } from "../../../src/models"; -import { Event, Organization, User } from "../../../src/models"; -import { connect, disconnect } from "../../helpers/db"; -import type { QueryUsersArgs } from "../../../src/types/generatedGraphQLTypes"; import type { Document } from "mongoose"; +import * as mongoose from "mongoose"; import { nanoid } from "nanoid"; +import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; import { BASE_URL, UNAUTHENTICATED_ERROR } from "../../../src/constants"; -import { beforeAll, afterAll, describe, it, expect, vi } from "vitest"; -import * as mongoose from "mongoose"; +import type { InterfaceUser } from "../../../src/models"; +import { AppUserProfile, Event, Organization, User } from "../../../src/models"; +import { users as usersResolver } from "../../../src/resolvers/Query/users"; +import type { QueryUsersArgs } from "../../../src/types/generatedGraphQLTypes"; +import { connect, disconnect } from "../../helpers/db"; import { createTestUser } from "../../helpers/user"; -let testUsers: (InterfaceUser & Document)[]; +let testUsers: (InterfaceUser & Document)[]; let MONGOOSE_INSTANCE: typeof mongoose; @@ -45,9 +45,9 @@ describe("resolvers -> Query -> users", () => { "../../../src/resolvers/Query/users" ); await mockedInProductionUserResolver?.({}, args, {}); - } catch (error: any) { + } catch (error: unknown) { expect(spy).toBeCalledWith(UNAUTHENTICATED_ERROR.MESSAGE); - expect(error.message).toEqual( + expect((error as Error).message).toEqual( `Translated ${UNAUTHENTICATED_ERROR.MESSAGE}` ); } @@ -88,44 +88,51 @@ describe("resolvers -> Query -> users", () => { password: "password", firstName: `firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid()}`, }, { email: `email${nanoid().toLowerCase()}@gmail.com`, password: "password", firstName: `firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid()}`, }, { email: `email${nanoid().toLowerCase()}@gmail.com`, password: "password", firstName: `firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid()}`, }, { email: `email${nanoid().toLowerCase()}@gmail.com`, password: "password", firstName: `firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid()}`, - userType: "SUPERADMIN", }, { email: `email${nanoid().toLowerCase()}@gmail.com`, password: "password", firstName: `firstName${nanoid()}`, lastName: `lastName${nanoid()}`, - appLanguageCode: `en${nanoid()}`, - userType: "ADMIN", }, ]); + const appUserProfiles = testUsers.map((user) => ({ + userId: user._id, + appLanguageCode: `en${nanoid()}`, + })); + + await AppUserProfile.insertMany(appUserProfiles); + await AppUserProfile.updateOne( + { + userId: testUsers[3]._id, + }, + { + isSuperAdmin: true, + } + ); const testOrganization = await Organization.create({ name: "name1", description: "description1", - isPublic: true, + userRegistrationRequired: false, creatorId: testUsers[0]._id, admins: [testUsers[0]._id], members: [testUsers[0]._id, testUsers[1]._id], @@ -158,12 +165,22 @@ describe("resolvers -> Query -> users", () => { }, { $push: { - createdOrganizations: testOrganization._id, - adminFor: testOrganization._id, joinedOrganizations: testOrganization._id, - createdEvents: testEvent._id, + registeredEvents: testEvent._id, + }, + } + ); + await AppUserProfile.updateOne( + { + userId: testUsers[0]._id, + }, + { + $push: { + adminFor: testOrganization._id, + createdOrganizations: testOrganization._id, eventAdmin: testEvent._id, + createdEvents: testEvent._id, }, } ); @@ -212,12 +229,8 @@ describe("resolvers -> Query -> users", () => { }) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); users = users.map((user) => ({ @@ -249,49 +262,8 @@ describe("resolvers -> Query -> users", () => { }) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") - .populate("joinedOrganizations") - .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") - .populate("organizationsBlockedBy") - .lean(); - - users = users.map((user) => ({ - ...user, - image: null, - })); - - expect(usersPayload).toEqual(users); - }); - - it(`returns populated array for organizationsBlockedBy fields when the client is a ADMIN`, async () => { - const args: QueryUsersArgs = { - where: { - id: testUsers[1].id, - }, - }; - - const sort = { - _id: 1, - }; - - const usersPayload = await usersResolver?.({}, args, { - userId: testUsers[4]._id, - }); - - let users = await User.find({ - _id: testUsers[1].id, - }) - .sort(sort) - .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .populate("organizationsBlockedBy") .lean(); @@ -305,16 +277,15 @@ describe("resolvers -> Query -> users", () => { it(`returns list of all existing users filtered by args.where === { id: testUsers[1].id, firstName: testUsers[1].firstName, - lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode }, args.adminApproved: true - and args.userType : "SUPERADMIN" and sorted by - args.orderBy === 'id_ASC'`, async () => { + lastName: testUsers[1].lastName, email: testUsers[1].email + }, args.adminApproved: true + and sorted by args.orderBy === 'id_ASC'`, async () => { const where = { _id: testUsers[1].id, firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }; const sort = { @@ -327,16 +298,16 @@ describe("resolvers -> Query -> users", () => { firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }, - userType: "SUPERADMIN", + // userType: "SUPERADMIN", adminApproved: true, orderBy: "id_ASC", }; const filterCriteria = { ...where, - userType: args.userType as string, + // userType: args.userType as string, adminApproved: args.adminApproved as boolean, }; @@ -347,12 +318,8 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(filterCriteria) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); users = users.map((user) => ({ @@ -365,16 +332,15 @@ describe("resolvers -> Query -> users", () => { it(`returns list of all existing users filtered by args.where === { id: testUsers[1].id, firstName: testUsers[1].firstName, - lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode }, args.adminApproved: false - and args.userType : "SUPERADMIN" and sorted by + lastName: testUsers[1].lastName, email: testUsers[1].email}, + args.adminApproved: false and sorted by args.orderBy === 'id_ASC'`, async () => { const where = { _id: testUsers[1].id, firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }; const sort = { @@ -387,16 +353,16 @@ describe("resolvers -> Query -> users", () => { firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }, - userType: "SUPERADMIN", + // userType: "SUPERADMIN", adminApproved: false, orderBy: "id_ASC", }; const filterCriteria = { ...where, - userType: args.userType as string, + // userType: args.userType as string, adminApproved: args.adminApproved as boolean, }; @@ -407,17 +373,15 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(filterCriteria) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + .populate("organizationsBlockedBy") .lean(); users = users.map((user) => ({ ...user, organizationsBlockedBy: [], + image: user.image ? `${BASE_URL}${user.image}` : null, })); expect(usersPayload).toEqual(users); @@ -425,15 +389,14 @@ describe("resolvers -> Query -> users", () => { it(`returns list of all existing users filtered by args.where === { id: testUsers[1].id, firstName: testUsers[1].firstName, - lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode } and sorted by + lastName: testUsers[1].lastName, email: testUsers[1].email} and sorted by args.orderBy === 'id_ASC'`, async () => { const where = { _id: testUsers[1].id, firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }; const sort = { @@ -446,7 +409,7 @@ describe("resolvers -> Query -> users", () => { firstName: testUsers[1].firstName, lastName: testUsers[1].lastName, email: testUsers[1].email, - appLanguageCode: testUsers[1].appLanguageCode, + // appLanguageCode: testUsers[1].appLanguageCode, }, orderBy: "id_ASC", }; @@ -460,12 +423,8 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); users = users.map((user) => ({ @@ -473,14 +432,13 @@ describe("resolvers -> Query -> users", () => { organizationsBlockedBy: [], image: user.image ? `${BASE_URL}${user.image}` : null, })); - + // console.log(usersPayload); expect(usersPayload).toEqual(users); }); it(`returns list of all existing users filtered by args.where === { id_not: testUsers[2]._id, firstName_not: testUsers[2].firstName, - lastName_not: testUsers[2].lastName, email_not: testUsers[2].email, - appLanguageCode_not: testUsers[2].appLanguageCode } and + lastName_not: testUsers[2].lastName, email_not: testUsers[2].email } and sorted by args.orderBy === 'id_Desc'`, async () => { const where = { _id: { @@ -495,9 +453,6 @@ describe("resolvers -> Query -> users", () => { email: { $ne: testUsers[2].email, }, - appLanguageCode: { - $ne: testUsers[2].appLanguageCode, - }, }; const sort = { @@ -506,11 +461,10 @@ describe("resolvers -> Query -> users", () => { const args: QueryUsersArgs = { where: { - id_not: testUsers[2]._id, + id_not: testUsers[2]._id.toString(), firstName_not: testUsers[2].firstName, lastName_not: testUsers[2].lastName, email_not: testUsers[2].email, - appLanguageCode_not: testUsers[2].appLanguageCode, }, orderBy: "id_DESC", }; @@ -525,12 +479,10 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + .lean(); users = users.map((user) => ({ @@ -544,8 +496,7 @@ describe("resolvers -> Query -> users", () => { it(`returns list of all existing users filtered by args.where === { id_in: [testUsers[1].id], firstName_in: [testUsers[1].firstName], - lastName_in: [testUsers[1].lastName], email_in: [testUsers[1].email], - appLanguageCode_in: [testUsers[1].appLanguageCode] } and + lastName_in: [testUsers[1].lastName], email_in: [testUsers[1].email] } and sorted by args.orderBy === 'firstName_ASC'`, async () => { const where = { _id: { @@ -560,9 +511,9 @@ describe("resolvers -> Query -> users", () => { email: { $in: [testUsers[1].email], }, - appLanguageCode: { - $in: [testUsers[1].appLanguageCode], - }, + // appLanguageCode: { + // $in: [testUsers[1].appLanguageCode], + // }, }; const sort = { @@ -575,7 +526,7 @@ describe("resolvers -> Query -> users", () => { firstName_in: [testUsers[1].firstName], lastName_in: [testUsers[1].lastName], email_in: [testUsers[1].email], - appLanguageCode_in: [testUsers[1].appLanguageCode], + // appLanguageCode_in: [testUsers[1].appLanguageCode], }, orderBy: "firstName_ASC", }; @@ -587,12 +538,8 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); users = users.map((user) => ({ @@ -622,9 +569,6 @@ describe("resolvers -> Query -> users", () => { email: { $nin: [testUsers[2].email], }, - appLanguageCode: { - $nin: [testUsers[2].appLanguageCode], - }, }; const sort = { @@ -633,11 +577,11 @@ describe("resolvers -> Query -> users", () => { const args: QueryUsersArgs = { where: { - id_not_in: [testUsers[2]._id], + id_not_in: [testUsers[2]._id.toString()], firstName_not_in: [testUsers[2].firstName], lastName_not_in: [testUsers[2].lastName], email_not_in: [testUsers[2].email], - appLanguageCode_not_in: [testUsers[2].appLanguageCode], + // appLanguageCode_not_in: [testUsers[2].appLanguageCode], }, orderBy: "firstName_DESC", }; @@ -652,12 +596,10 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + .lean(); users = users.map((user) => ({ @@ -687,10 +629,10 @@ describe("resolvers -> Query -> users", () => { $regex: testUsers[1].email, $options: "i", }, - appLanguageCode: { - $regex: testUsers[1].appLanguageCode, - $options: "i", - }, + // appLanguageCode: { + // $regex: testUsers[1].appLanguageCode, + // $options: "i", + // }, }; const sort = { @@ -702,7 +644,7 @@ describe("resolvers -> Query -> users", () => { firstName_contains: testUsers[1].firstName, lastName_contains: testUsers[1].lastName, email_contains: testUsers[1].email, - appLanguageCode_contains: testUsers[1].appLanguageCode, + // appLanguageCode_contains: testUsers[1].appLanguageCode, }, orderBy: "lastName_ASC", }; @@ -717,12 +659,10 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + .lean(); users = users.map((user) => ({ @@ -743,7 +683,7 @@ describe("resolvers -> Query -> users", () => { firstName: new RegExp("^" + testUsers[1].firstName), lastName: new RegExp("^" + testUsers[1].lastName), email: new RegExp("^" + testUsers[1].email), - appLanguageCode: new RegExp("^" + testUsers[1].appLanguageCode), + // appLanguageCode: new RegExp("^" + testUsers[1].appLanguageCode), }; const sort = { @@ -755,7 +695,7 @@ describe("resolvers -> Query -> users", () => { firstName_starts_with: testUsers[1].firstName, lastName_starts_with: testUsers[1].lastName, email_starts_with: testUsers[1].email, - appLanguageCode_starts_with: testUsers[1].appLanguageCode, + // appLanguageCode_starts_with: testUsers[1].appLanguageCode, }, orderBy: "lastName_DESC", }; @@ -769,92 +709,10 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") - .populate("joinedOrganizations") - .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") - .lean(); - - users = users.map((user) => ({ - ...user, - organizationsBlockedBy: [], - image: user.image ? `${BASE_URL}${user.image}` : null, - })); - - expect(usersPayload).toEqual(users); - }); - - it(`returns list of all existing users sorted by - args.orderBy === 'appLanguageCode_ASC'`, async () => { - const where = {}; - - const sort = { - appLanguageCode: 1, - }; - - const args: QueryUsersArgs = { - where: null, - orderBy: "appLanguageCode_ASC", - }; - - const context = { - apiRootUrl: BASE_URL, - userId: testUsers[0]._id, - }; - - const usersPayload = await usersResolver?.({}, args, context); - let users = await User.find(where) - .sort(sort) - .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") - .lean(); - - users = users.map((user) => ({ - ...user, - organizationsBlockedBy: [], - image: user.image ? `${BASE_URL}${user.image}` : null, - })); - - expect(usersPayload).toEqual(users); - }); - - it(`returns list of all existing users sorted by - args.orderBy === 'appLanguageCode_DESC'`, async () => { - const where = {}; - - const sort = { - appLanguageCode: -1, - }; - - const args: QueryUsersArgs = { - where: null, - orderBy: "appLanguageCode_DESC", - }; - - const context = { - apiRootUrl: BASE_URL, - userId: testUsers[0]._id, - }; - - const usersPayload = await usersResolver?.({}, args, context); - let users = await User.find(where) - .sort(sort) - .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") - .populate("joinedOrganizations") - .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); users = users.map((user) => ({ @@ -889,12 +747,10 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + .lean(); users = users.map((user) => ({ @@ -978,12 +834,10 @@ describe("resolvers -> Query -> users", () => { let users = await User.find(where) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + .lean(); users = users.map((user) => ({ @@ -994,4 +848,34 @@ describe("resolvers -> Query -> users", () => { expect(usersPayload).toEqual(users); }); + it("throws error if user does not have appUserProfile", async () => { + const { requestContext } = await import("../../../src/libraries"); + + const spy = vi + .spyOn(requestContext, "translate") + .mockImplementationOnce((message) => `Translated ${message}`); + await AppUserProfile.deleteOne({ + userId: testUsers[0]._id, + }); + const args: QueryUsersArgs = { + orderBy: null, + where: { + id: testUsers[0].id, + }, + }; + const context = { + userId: testUsers[0]._id, + }; + try { + const { users: mockedInProductionUserResolver } = await import( + "../../../src/resolvers/Query/users" + ); + await mockedInProductionUserResolver?.({}, args, context); + } catch (error: unknown) { + expect(spy).toBeCalledWith(UNAUTHENTICATED_ERROR.MESSAGE); + expect((error as Error).message).toEqual( + `Translated ${UNAUTHENTICATED_ERROR.MESSAGE}` + ); + } + }); }); diff --git a/tests/resolvers/Query/usersConnection.spec.ts b/tests/resolvers/Query/usersConnection.spec.ts index 240c4e3c9a..b0166b8808 100644 --- a/tests/resolvers/Query/usersConnection.spec.ts +++ b/tests/resolvers/Query/usersConnection.spec.ts @@ -1,16 +1,16 @@ import "dotenv/config"; -import { usersConnection as usersConnectionResolver } from "../../../src/resolvers/Query/usersConnection"; -import { connect, disconnect } from "../../helpers/db"; import type mongoose from "mongoose"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import { User } from "../../../src/models"; +import { usersConnection as usersConnectionResolver } from "../../../src/resolvers/Query/usersConnection"; import type { QueryUsersConnectionArgs } from "../../../src/types/generatedGraphQLTypes"; -import { beforeAll, afterAll, describe, it, expect } from "vitest"; +import { connect, disconnect } from "../../helpers/db"; +import { createEventWithRegistrant } from "../../helpers/events"; import type { TestUserType } from "../../helpers/userAndOrg"; import { - createTestUserAndOrganization, createTestUser, + createTestUserAndOrganization, } from "../../helpers/userAndOrg"; -import { createEventWithRegistrant } from "../../helpers/events"; let MONGOOSE_INSTANCE: typeof mongoose; let testUsers: TestUserType[]; @@ -47,12 +47,8 @@ describe("resolvers -> Query -> usersConnection", () => { .limit(0) .skip(0) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -68,7 +64,6 @@ describe("resolvers -> Query -> usersConnection", () => { firstName: testUsers[1]?.firstName, lastName: testUsers[1]?.lastName, email: testUsers[1]?.email, - appLanguageCode: testUsers[1]?.appLanguageCode, }; const sort = { @@ -83,7 +78,6 @@ describe("resolvers -> Query -> usersConnection", () => { firstName: testUsers[1]?.firstName, lastName: testUsers[1]?.lastName, email: testUsers[1]?.email, - appLanguageCode: testUsers[1]?.appLanguageCode, }, orderBy: "id_ASC", }; @@ -99,12 +93,8 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -128,9 +118,9 @@ describe("resolvers -> Query -> usersConnection", () => { email: { $ne: testUsers[2]?.email, }, - appLanguageCode: { - $ne: testUsers[2]?.appLanguageCode, - }, + // appLanguageCode: { + // $ne: testUsers[2]?.appLanguageCode, + // }, }; const sort = { @@ -145,7 +135,7 @@ describe("resolvers -> Query -> usersConnection", () => { firstName_not: testUsers[2]?.firstName, lastName_not: testUsers[2]?.lastName, email_not: testUsers[2]?.email, - appLanguageCode_not: testUsers[2]?.appLanguageCode, + // appLanguageCode_not: testUsers[2]?.appLanguageCode, }, orderBy: "id_DESC", }; @@ -161,12 +151,8 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -179,20 +165,20 @@ describe("resolvers -> Query -> usersConnection", () => { sorted by args.orderBy === 'firstName_ASC'`, async () => { const where = { _id: { - $in: [testUsers[1]!.id], + $in: [testUsers[1]?.id ?? ""], }, firstName: { - $in: [testUsers[1]!.firstName], + $in: [testUsers[1]?.firstName ?? ""], }, lastName: { - $in: [testUsers[1]!.lastName], + $in: [testUsers[1]?.lastName ?? ""], }, email: { - $in: [testUsers[1]!.email], - }, - appLanguageCode: { - $in: [testUsers[1]!.appLanguageCode], + $in: [testUsers[1]?.email ?? ""], }, + // appLanguageCode: { + // $in: [testUsers[1]?.appLanguageCode], + // }, }; const sort = { @@ -204,10 +190,10 @@ describe("resolvers -> Query -> usersConnection", () => { skip: 1, where: { id_in: [testUsers[1]?.id], - firstName_in: [testUsers[1]!.firstName], - lastName_in: [testUsers[1]!.lastName], - email_in: [testUsers[1]!.email], - appLanguageCode_in: [testUsers[1]!.appLanguageCode], + firstName_in: [testUsers[1]?.firstName ?? ""], + lastName_in: [testUsers[1]?.lastName ?? ""], + email_in: [testUsers[1]?.email], + // appLanguageCode_in: [testUsers[1]!.appLanguageCode], }, orderBy: "firstName_ASC", }; @@ -223,12 +209,9 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -241,20 +224,20 @@ describe("resolvers -> Query -> usersConnection", () => { sorted by args.orderBy === 'firstName_DESC'`, async () => { const where = { _id: { - $nin: [testUsers[2]!._id], + $nin: [testUsers[2]?._id ?? ""], }, firstName: { - $nin: [testUsers[2]!.firstName], + $nin: [testUsers[2]?.firstName ?? ""], }, lastName: { - $nin: [testUsers[2]!.lastName], + $nin: [testUsers[2]?.lastName ?? ""], }, email: { - $nin: [testUsers[2]!.email], - }, - appLanguageCode: { - $nin: [testUsers[2]!.appLanguageCode], + $nin: [testUsers[2]?.email ?? ""], }, + // appLanguageCode: { + // $nin: [testUsers[2]!.appLanguageCode], + // }, }; const sort = { @@ -266,10 +249,10 @@ describe("resolvers -> Query -> usersConnection", () => { skip: 1, where: { id_not_in: [testUsers[2]?._id], - firstName_not_in: [testUsers[2]!.firstName], - lastName_not_in: [testUsers[2]!.lastName], - email_not_in: [testUsers[2]!.email], - appLanguageCode_not_in: [testUsers[2]!.appLanguageCode], + firstName_not_in: [testUsers[2]?.firstName ?? ""], + lastName_not_in: [testUsers[2]?.lastName ?? ""], + email_not_in: [testUsers[2]?.email ?? ""], + // appLanguageCode_not_in: [testUsers[2]?.appLanguageCode], }, orderBy: "firstName_DESC", }; @@ -285,12 +268,8 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -314,10 +293,10 @@ describe("resolvers -> Query -> usersConnection", () => { $regex: testUsers[1]?.email, $options: "i", }, - appLanguageCode: { - $regex: testUsers[1]?.appLanguageCode, - $options: "i", - }, + // appLanguageCode: { + // $regex: testUsers[1]?.appLanguageCode, + // $options: "i", + // }, }; const sort = { @@ -331,7 +310,7 @@ describe("resolvers -> Query -> usersConnection", () => { firstName_contains: testUsers[1]?.firstName, lastName_contains: testUsers[1]?.lastName, email_contains: testUsers[1]?.email, - appLanguageCode_contains: testUsers[1]?.appLanguageCode, + // appLanguageCode_contains: testUsers[1]?.appLanguageCode, }, orderBy: "lastName_ASC", }; @@ -347,12 +326,10 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") + .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") + .lean(); expect(usersConnectionPayload).toEqual(users); @@ -367,7 +344,7 @@ describe("resolvers -> Query -> usersConnection", () => { firstName: new RegExp("^" + testUsers[1]?.firstName), lastName: new RegExp("^" + testUsers[1]?.lastName), email: new RegExp("^" + testUsers[1]?.email), - appLanguageCode: new RegExp("^" + testUsers[1]?.appLanguageCode), + // appLanguageCode: new RegExp("^" + testUsers[1]?.appLanguageCode), }; const sort = { @@ -381,7 +358,7 @@ describe("resolvers -> Query -> usersConnection", () => { firstName_starts_with: testUsers[1]?.firstName, lastName_starts_with: testUsers[1]?.lastName, email_starts_with: testUsers[1]?.email, - appLanguageCode_starts_with: testUsers[1]?.appLanguageCode, + // appLanguageCode_starts_with: testUsers[1]?.appLanguageCode, }, orderBy: "lastName_DESC", }; @@ -397,86 +374,8 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") - .populate("joinedOrganizations") - .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") - .lean(); - - expect(usersConnectionPayload).toEqual(users); - }); - - it(`returns paginated list of users sorted by - args.orderBy === 'appLanguageCode_ASC'`, async () => { - const where = {}; - - const sort = { - appLanguageCode: 1, - }; - - const args: QueryUsersConnectionArgs = { - first: 2, - skip: 1, - where: null, - orderBy: "appLanguageCode_ASC", - }; - - const usersConnectionPayload = await usersConnectionResolver?.( - {}, - args, - {} - ); - - const users = await User.find(where) - .limit(2) - .skip(1) - .sort(sort) - .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") - .populate("joinedOrganizations") - .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") - .lean(); - - expect(usersConnectionPayload).toEqual(users); - }); - - it(`returns paginated list of users sorted by - args.orderBy === 'appLanguageCode_DESC'`, async () => { - const where = {}; - - const sort = { - appLanguageCode: -1, - }; - - const args: QueryUsersConnectionArgs = { - first: 2, - skip: 1, - where: null, - orderBy: "appLanguageCode_DESC", - }; - - const usersConnectionPayload = await usersConnectionResolver?.( - {}, - args, - {} - ); - - const users = await User.find(where) - .limit(2) - .skip(1) - .sort(sort) - .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -508,12 +407,8 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -545,12 +440,8 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); @@ -579,12 +470,8 @@ describe("resolvers -> Query -> usersConnection", () => { .skip(1) .sort(sort) .select(["-password"]) - .populate("createdOrganizations") - .populate("createdEvents") .populate("joinedOrganizations") .populate("registeredEvents") - .populate("eventAdmin") - .populate("adminFor") .lean(); expect(usersConnectionPayload).toEqual(users); diff --git a/tests/utilities/adminCheck.spec.ts b/tests/utilities/adminCheck.spec.ts index 674d72042d..e08b24f2aa 100644 --- a/tests/utilities/adminCheck.spec.ts +++ b/tests/utilities/adminCheck.spec.ts @@ -1,4 +1,5 @@ import "dotenv/config"; +import mongoose from "mongoose"; import { afterAll, afterEach, @@ -8,14 +9,13 @@ import { it, vi, } from "vitest"; -import { connect, disconnect } from "../helpers/db"; import { USER_NOT_AUTHORIZED_ADMIN } from "../../src/constants"; +import { ApplicationError } from "../../src/libraries/errors"; +import type { InterfaceOrganization } from "../../src/models"; +import { AppUserProfile, Organization } from "../../src/models"; +import { connect, disconnect } from "../helpers/db"; import type { TestOrganizationType, TestUserType } from "../helpers/userAndOrg"; import { createTestUserAndOrganization } from "../helpers/userAndOrg"; -import mongoose from "mongoose"; -import type { InterfaceOrganization } from "../../src/models"; -import { Organization, User } from "../../src/models"; -import { ApplicationError } from "../../src/libraries/errors"; let testUser: TestUserType; let testOrganization: TestOrganizationType; @@ -50,8 +50,8 @@ describe("utilities -> adminCheck", () => { testUser?._id, testOrganization ?? ({} as InterfaceOrganization) ); - } catch (error: any) { - expect(error.message).toEqual( + } catch (error: unknown) { + expect((error as Error).message).toEqual( `Translated ${USER_NOT_AUTHORIZED_ADMIN.MESSAGE}` ); } @@ -59,12 +59,12 @@ describe("utilities -> adminCheck", () => { }); it("throws no error if userIsOrganizationAdmin === false and isUserSuperAdmin === true", async () => { - const updatedUser = await User.findOneAndUpdate( + const updatedUser = await AppUserProfile.findOneAndUpdate( { - _id: testUser?._id, + userId: testUser?._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, }, { new: true, @@ -76,26 +76,13 @@ describe("utilities -> adminCheck", () => { await expect( adminCheck( - updatedUser?._id, + updatedUser?.userId?.toString() ?? "", testOrganization ?? ({} as InterfaceOrganization) ) ).resolves.not.toThrowError(); }); it("throws no error if user is an admin in that organization but not super admin", async () => { - const updatedUser = await User.findOneAndUpdate( - { - _id: testUser?._id, - }, - { - userType: "USER", - }, - { - new: true, - upsert: true, - } - ); - const updatedOrganization = await Organization.findOneAndUpdate( { _id: testOrganization?._id, @@ -115,7 +102,7 @@ describe("utilities -> adminCheck", () => { await expect( adminCheck( - updatedUser?._id, + testUser?._id, updatedOrganization ?? ({} as InterfaceOrganization) ) ).resolves.not.toThrowError(); diff --git a/tests/utilities/auth.spec.ts b/tests/utilities/auth.spec.ts index 8c6484844b..7053ec4a05 100644 --- a/tests/utilities/auth.spec.ts +++ b/tests/utilities/auth.spec.ts @@ -1,18 +1,20 @@ -import { expect, describe, it, beforeAll, afterAll } from "vitest"; +import jwt from "jsonwebtoken"; import type mongoose from "mongoose"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; +import type { InterfaceAppUserProfile, InterfaceUser } from "../../src/models"; +import { AppUserProfile } from "../../src/models"; import { createAccessToken, createRefreshToken, revokeRefreshToken, } from "../../src/utilities"; -import type { InterfaceUser } from "../../src/models"; -import { User } from "../../src/models"; -import jwt from "jsonwebtoken"; +import { connect, disconnect } from "../helpers/db"; import type { TestUserType } from "../helpers/user"; import { createTestUserFunc } from "../helpers/user"; -import { connect, disconnect } from "../helpers/db"; +import type { TestAppUserProfileType } from "../helpers/userAndOrg"; let user: TestUserType; +let appUserProfile: TestAppUserProfileType; let MONGOOSE_INSTANCE: typeof mongoose; export interface InterfaceJwtTokenPayload { tokenVersion: number; @@ -25,6 +27,9 @@ export interface InterfaceJwtTokenPayload { beforeAll(async () => { MONGOOSE_INSTANCE = await connect(); user = await createTestUserFunc(); + appUserProfile = await AppUserProfile.findOne({ + userId: user?._id, + }); }); afterAll(async () => { @@ -34,7 +39,10 @@ afterAll(async () => { describe("createAccessToken", () => { it("should create a JWT token with the correct payload", async () => { const token = createAccessToken( - user ? user.toObject() : ({} as InterfaceUser) + user ? user.toObject() : ({} as InterfaceUser), + appUserProfile + ? appUserProfile.toObject() + : ({} as InterfaceAppUserProfile) ); expect(token).toBeDefined(); @@ -43,7 +51,7 @@ describe("createAccessToken", () => { expect(decodedToken).not.toBeNull(); expect((decodedToken as InterfaceJwtTokenPayload).tokenVersion).toBe( - user?.tokenVersion + appUserProfile?.tokenVersion ); expect((decodedToken as InterfaceJwtTokenPayload).userId).toBe( user && user._id ? user._id.toString() : undefined @@ -61,7 +69,10 @@ describe("createAccessToken", () => { describe("createRefreshToken", () => { it("should create a JWT token with the correct payload", () => { const token = createRefreshToken( - user ? user.toObject() : ({} as InterfaceUser) + user ? user.toObject() : ({} as InterfaceUser), + appUserProfile + ? appUserProfile.toObject() + : ({} as InterfaceAppUserProfile) ); expect(token).toBeDefined(); @@ -69,7 +80,7 @@ describe("createRefreshToken", () => { const decodedToken = jwt.decode(token); expect((decodedToken as InterfaceJwtTokenPayload).tokenVersion).toBe( - user?.tokenVersion + appUserProfile?.tokenVersion ); expect((decodedToken as InterfaceJwtTokenPayload).userId).toBe( user && user._id ? user._id.toString() : undefined @@ -86,10 +97,13 @@ describe("createRefreshToken", () => { describe("revokeRefreshToken", () => { it("should unset the token field in the user document", async () => { - await revokeRefreshToken(user?._id); + await revokeRefreshToken(user?._id.toString() ?? ""); - const updatedUser = await User.findOne({ _id: user?._id }); + // const updatedUser = await User.findOne({ _id: user?._id }); + const updateAppUserProfile = await AppUserProfile.findOne({ + userId: user?._id, + }); - expect(updatedUser?.token).toBeUndefined(); + expect(updateAppUserProfile?.token).toBeUndefined(); }); }); diff --git a/tests/utilities/createSampleOrganizationUtil.spec.ts b/tests/utilities/createSampleOrganizationUtil.spec.ts index 54d614e23e..2c65e08ac4 100644 --- a/tests/utilities/createSampleOrganizationUtil.spec.ts +++ b/tests/utilities/createSampleOrganizationUtil.spec.ts @@ -1,12 +1,12 @@ -import { describe, it, expect, beforeAll, afterAll } from "vitest"; +import { faker } from "@faker-js/faker"; import type mongoose from "mongoose"; +import { afterAll, beforeAll, describe, expect, it } from "vitest"; import { Plugin } from "../../src/models"; -import { faker } from "@faker-js/faker"; import { + generateEventData as generateEventDataFn, + generatePostData, generateRandomPlugins, generateUserData, - generatePostData, - generateEventData as generateEventDataFn, } from "../../src/utilities/createSampleOrganizationUtil"; import { connect, disconnect } from "../helpers/db"; @@ -25,7 +25,9 @@ describe("generateUserData function", () => { it("should return correct properties given ADMIN userType and organization Id", async () => { const organizationId = faker.database.mongodbObjectId(); - const user = await generateUserData(organizationId, "ADMIN"); + const userData = await generateUserData(organizationId, "ADMIN"); + const user = userData.user; + const appUserProfile = userData.appUserProfile; expect(typeof user._id.toString()).toBe("string"); expect(typeof user.firstName).toBe("string"); @@ -37,18 +39,17 @@ describe("generateUserData function", () => { expect(user.joinedOrganizations.length).toBe(1); expect(typeof user.joinedOrganizations[0].toString()).toBe("string"); - expect(typeof user.userType).toBe("string"); - expect(user.userType).toBe("ADMIN"); - - expect(Array.isArray(user.adminFor)).toBe(true); - expect(user.adminFor.length).toBe(1); - expect(typeof user.adminFor[0].toString()).toBe("string"); + expect(Array.isArray(appUserProfile.adminFor)).toBe(true); + expect(appUserProfile.adminFor?.length).toBe(1); + expect(typeof appUserProfile.adminFor?.[0]?.toString()).toBe("string"); }); it("should return User with correct properties", async () => { const organizationId = faker.database.mongodbObjectId(); - const user = await generateUserData(organizationId, "SUPERADMIN"); + const userData = await generateUserData(organizationId, "SUPERADMIN"); + const user = userData.user; + const appUserProfile = userData.appUserProfile; expect(typeof user._id.toString()).toBe("string"); expect(typeof user.firstName).toBe("string"); @@ -60,18 +61,17 @@ describe("generateUserData function", () => { expect(user.joinedOrganizations.length).toBe(1); expect(typeof user.joinedOrganizations[0].toString()).toBe("string"); - expect(typeof user.userType).toBe("string"); - expect(user.userType).toBe("SUPERADMIN"); + expect(appUserProfile.isSuperAdmin).toBe(true); - expect(Array.isArray(user.adminFor)).toBe(true); - expect(user.adminFor.length).toBe(0); + expect(Array.isArray(appUserProfile.adminFor)).toBe(true); + expect(appUserProfile.adminFor.length).toBe(0); }); it("should return Event with correct properties", async () => { - const users = [ + const userData = [ await generateUserData(faker.database.mongodbObjectId(), "USER"), ]; - + const users = userData.map((user) => user.user); const organizationId = faker.database.mongodbObjectId(); const event = await generateEventDataFn(users, organizationId); expect(event._id).toEqual(expect.any(Object)); @@ -100,9 +100,10 @@ describe("generateUserData function", () => { describe("generatePostData function", () => { it("should return Post with the correct properties", async () => { - const users = [ + const usersData = [ await generateUserData(faker.database.mongodbObjectId(), "USER"), ]; + const users = usersData.map((user) => user.user); const organizationId = faker.database.mongodbObjectId(); // Fake ObjectID for the organization const post = await generatePostData(users, organizationId); @@ -124,9 +125,10 @@ describe("generatePostData function", () => { describe("generateRandomPlugins function", () => { it("should generate and save the specified number of plugins with correct properties", async () => { const numberOfPlugins = 5; - const users = [ + const usersData = [ await generateUserData(faker.database.mongodbObjectId(), "USER"), ]; + const users = usersData.map((user) => user.user); const pluginPromises = await generateRandomPlugins( numberOfPlugins, @@ -134,9 +136,9 @@ describe("generatePostData function", () => { ); expect(Array.isArray(pluginPromises)).toBe(true); - expect(pluginPromises!.length).toBe(numberOfPlugins); + expect(pluginPromises.length).toBe(numberOfPlugins); - await Promise.all(pluginPromises!); + await Promise.all(pluginPromises); const plugins = await Plugin.find(); expect(plugins.length).toBe(numberOfPlugins); diff --git a/tests/utilities/superAdminCheck.spec.ts b/tests/utilities/superAdminCheck.spec.ts index 780237efbd..768f1b55dd 100644 --- a/tests/utilities/superAdminCheck.spec.ts +++ b/tests/utilities/superAdminCheck.spec.ts @@ -8,16 +8,22 @@ import { it, vi, } from "vitest"; -import { connect, disconnect } from "../../src/db"; import { USER_NOT_AUTHORIZED_SUPERADMIN } from "../../src/constants"; -import type { TestUserType } from "../helpers/userAndOrg"; +import { connect, disconnect } from "../../src/db"; +import { AppUserProfile } from "../../src/models"; import { createTestUserFunc } from "../helpers/user"; +import type { + TestAppUserProfileType, + TestUserType, +} from "../helpers/userAndOrg"; let testUser: TestUserType; +let testAppUserProfile: TestAppUserProfileType; beforeAll(async () => { connect(); testUser = await createTestUserFunc(); + testAppUserProfile = await AppUserProfile.findOne({ userId: testUser?._id }); }); afterAll(async () => { @@ -39,8 +45,8 @@ describe("utilities -> superAdminCheck", () => { try { const { superAdminCheck } = await import("../../src/utilities"); - if (testUser) { - superAdminCheck(testUser); + if (testAppUserProfile) { + superAdminCheck(testAppUserProfile); } } catch (error: unknown) { if (!(error instanceof Error)) return; diff --git a/tests/utilities/userFamilyAdminCheck.spec.ts b/tests/utilities/userFamilyAdminCheck.spec.ts index fa7d3c45a8..468c5075e2 100644 --- a/tests/utilities/userFamilyAdminCheck.spec.ts +++ b/tests/utilities/userFamilyAdminCheck.spec.ts @@ -1,4 +1,5 @@ import "dotenv/config"; +import mongoose from "mongoose"; import { afterAll, afterEach, @@ -8,18 +9,17 @@ import { it, vi, } from "vitest"; -import { connect, disconnect } from "../helpers/db"; import { USER_NOT_AUTHORIZED_ADMIN } from "../../src/constants"; +import { AppUserProfile } from "../../src/models"; +import type { InterfaceUserFamily } from "../../src/models/userFamily"; +import { UserFamily } from "../../src/models/userFamily"; +import { connect, disconnect } from "../helpers/db"; +import { createTestUserFunc } from "../helpers/user"; import type { TestUserFamilyType, TestUserType, } from "../helpers/userAndUserFamily"; import { createTestUserAndUserFamily } from "../helpers/userAndUserFamily"; -import { createTestUserFunc } from "../helpers/user"; -import mongoose from "mongoose"; -import type { InterfaceUserFamily } from "../../src/models/userFamily"; -import { User } from "../../src/models"; -import { UserFamily } from "../../src/models/userFamily"; let testUser: TestUserType; let testUserFamily: TestUserFamilyType; @@ -65,12 +65,12 @@ describe("utilities -> userFamilyAdminCheck", () => { }); it("throws no error if userIsUserFamilyAdmin === false and isUserSuperAdmin === true", async () => { - const updatedUser = await User.findOneAndUpdate( + const updatedUser = await AppUserProfile.findOneAndUpdate( { - _id: testUser?._id, + userId: testUser?._id, }, { - userType: "SUPERADMIN", + isSuperAdmin: true, }, { new: true, @@ -84,26 +84,13 @@ describe("utilities -> userFamilyAdminCheck", () => { await expect( adminCheck( - updatedUser?._id, + updatedUser?.userId?.toString() ?? "", testUserFamily ?? ({} as InterfaceUserFamily) ) ).resolves.not.toThrowError(); }); it("throws no error if user is an admin in that user family but not super admin", async () => { - const updatedUser = await User.findOneAndUpdate( - { - _id: testUser?._id, - }, - { - userType: "USER", - }, - { - new: true, - upsert: true, - } - ); - const updatedUserFamily = await UserFamily.findOneAndUpdate( { _id: testUserFamily?._id, @@ -125,7 +112,7 @@ describe("utilities -> userFamilyAdminCheck", () => { await expect( adminCheck( - updatedUser?._id, + testUser?._id, updatedUserFamily ?? ({} as InterfaceUserFamily) ) ).resolves.not.toThrowError();