|
1 | 1 | import { CR_Type, Organization, Scope_CR_Why_Type, User, WBS_Element_Status } from '@prisma/client'; |
2 | 2 | import { createTestOrganization, createTestUser, resetUsers } from '../test-utils'; |
3 | 3 | import ChangeRequestsService from '../../src/services/change-requests.services'; |
4 | | -import { supermanAdmin } from '../test-data/users.test-data'; |
| 4 | +import { supermanAdmin, aquamanLeadership, greenlanternHead, flashAdmin, robinMember } from '../test-data/users.test-data'; |
5 | 5 | import { ProjectProposedChangesCreateArgs, WorkPackageProposedChangesCreateArgs } from 'shared'; |
6 | 6 | import prisma from '../../src/prisma/prisma'; |
| 7 | +import { AccessDeniedException } from '../../src/utils/errors.utils'; |
7 | 8 |
|
8 | 9 | describe('Change Request Tests', () => { |
9 | 10 | let orgId: string; |
@@ -284,4 +285,207 @@ describe('Change Request Tests', () => { |
284 | 285 | expect(wbsElement?.workPackageNumber).toEqual(14); |
285 | 286 | }); |
286 | 287 | }); |
| 288 | + |
| 289 | + describe('Review Change Request with Requested Reviewers', () => { |
| 290 | + let submitterUser: User; |
| 291 | + let leadershipUser1: User; |
| 292 | + let leadershipUser2: User; |
| 293 | + let nonRequestedLeadership: User; |
| 294 | + let memberUser: User; |
| 295 | + let changeRequestId: string; |
| 296 | + |
| 297 | + beforeEach(async () => { |
| 298 | + // Use the existing user from the main beforeEach |
| 299 | + submitterUser = user; |
| 300 | + |
| 301 | + // Create users with User_Settings that include slackId (needed for requestCRReview) |
| 302 | + leadershipUser1 = await createTestUser(aquamanLeadership, orgId, { |
| 303 | + id: 'aquaman-settings', |
| 304 | + userId: '', |
| 305 | + defaultTheme: 'DARK' as any, |
| 306 | + slackId: 'slack-aquaman' |
| 307 | + }); |
| 308 | + |
| 309 | + leadershipUser2 = await createTestUser(greenlanternHead, orgId, { |
| 310 | + id: 'greenlantern-settings', |
| 311 | + userId: '', |
| 312 | + defaultTheme: 'DARK' as any, |
| 313 | + slackId: 'slack-greenlantern' |
| 314 | + }); |
| 315 | + |
| 316 | + nonRequestedLeadership = await createTestUser(flashAdmin, orgId, { |
| 317 | + id: 'flash-settings', |
| 318 | + userId: '', |
| 319 | + defaultTheme: 'DARK' as any, |
| 320 | + slackId: 'slack-flash' |
| 321 | + }); |
| 322 | + |
| 323 | + memberUser = await createTestUser(robinMember, orgId); |
| 324 | + |
| 325 | + // Create a simple change request with a proposed solution |
| 326 | + const cr = await ChangeRequestsService.createStandardChangeRequest( |
| 327 | + submitterUser, |
| 328 | + 12, |
| 329 | + 13, |
| 330 | + 14, |
| 331 | + CR_Type.ISSUE, |
| 332 | + 'What is being changed', |
| 333 | + [{ type: Scope_CR_Why_Type.COMPETITION, explain: 'Why it is being changed' }], |
| 334 | + [ |
| 335 | + { |
| 336 | + description: 'Proposed solution', |
| 337 | + scopeImpact: 'Low impact', |
| 338 | + timelineImpact: 0, |
| 339 | + budgetImpact: 0 |
| 340 | + } |
| 341 | + ], |
| 342 | + organization, |
| 343 | + null, |
| 344 | + null |
| 345 | + ); |
| 346 | + |
| 347 | + changeRequestId = cr.crId; |
| 348 | + }); |
| 349 | + |
| 350 | + it('allows any leadership to review when no reviewers are requested', async () => { |
| 351 | + const reviewResult = await ChangeRequestsService.reviewChangeRequest( |
| 352 | + nonRequestedLeadership, |
| 353 | + changeRequestId, |
| 354 | + 'Looks good', |
| 355 | + false, |
| 356 | + organization, |
| 357 | + null |
| 358 | + ); |
| 359 | + |
| 360 | + expect(reviewResult).toBe(changeRequestId); |
| 361 | + |
| 362 | + const updatedCR = await prisma.change_Request.findUnique({ |
| 363 | + where: { crId: changeRequestId } |
| 364 | + }); |
| 365 | + |
| 366 | + expect(updatedCR?.reviewerId).toBe(nonRequestedLeadership.userId); |
| 367 | + expect(updatedCR?.accepted).toBe(false); |
| 368 | + }); |
| 369 | + |
| 370 | + it('allows requested reviewer to review when reviewers are requested', async () => { |
| 371 | + await ChangeRequestsService.requestCRReview( |
| 372 | + submitterUser, |
| 373 | + [leadershipUser1.userId, leadershipUser2.userId], |
| 374 | + changeRequestId, |
| 375 | + organization |
| 376 | + ); |
| 377 | + |
| 378 | + const reviewResult = await ChangeRequestsService.reviewChangeRequest( |
| 379 | + leadershipUser1, |
| 380 | + changeRequestId, |
| 381 | + 'Approved', |
| 382 | + false, |
| 383 | + organization, |
| 384 | + null |
| 385 | + ); |
| 386 | + |
| 387 | + expect(reviewResult).toBe(changeRequestId); |
| 388 | + |
| 389 | + const updatedCR = await prisma.change_Request.findUnique({ |
| 390 | + where: { crId: changeRequestId } |
| 391 | + }); |
| 392 | + |
| 393 | + expect(updatedCR?.reviewerId).toBe(leadershipUser1.userId); |
| 394 | + expect(updatedCR?.accepted).toBe(false); |
| 395 | + }); |
| 396 | + |
| 397 | + it('rejects non-requested leadership when reviewers are requested', async () => { |
| 398 | + await ChangeRequestsService.requestCRReview( |
| 399 | + submitterUser, |
| 400 | + [leadershipUser1.userId, leadershipUser2.userId], |
| 401 | + changeRequestId, |
| 402 | + organization |
| 403 | + ); |
| 404 | + |
| 405 | + await expect( |
| 406 | + ChangeRequestsService.reviewChangeRequest( |
| 407 | + nonRequestedLeadership, |
| 408 | + changeRequestId, |
| 409 | + 'I want to review this', |
| 410 | + true, |
| 411 | + organization, |
| 412 | + null |
| 413 | + ) |
| 414 | + ).rejects.toThrow(AccessDeniedException); |
| 415 | + |
| 416 | + await expect( |
| 417 | + ChangeRequestsService.reviewChangeRequest( |
| 418 | + nonRequestedLeadership, |
| 419 | + changeRequestId, |
| 420 | + 'I want to review this', |
| 421 | + true, |
| 422 | + organization, |
| 423 | + null |
| 424 | + ) |
| 425 | + ).rejects.toThrow('Only requested reviewers can review this change request!'); |
| 426 | + }); |
| 427 | + |
| 428 | + it('allows second requested reviewer to review when reviewers are requested', async () => { |
| 429 | + await ChangeRequestsService.requestCRReview( |
| 430 | + submitterUser, |
| 431 | + [leadershipUser1.userId, leadershipUser2.userId], |
| 432 | + changeRequestId, |
| 433 | + organization |
| 434 | + ); |
| 435 | + |
| 436 | + const reviewResult = await ChangeRequestsService.reviewChangeRequest( |
| 437 | + leadershipUser2, |
| 438 | + changeRequestId, |
| 439 | + 'Approved by second reviewer', |
| 440 | + false, |
| 441 | + organization, |
| 442 | + null |
| 443 | + ); |
| 444 | + |
| 445 | + expect(reviewResult).toBe(changeRequestId); |
| 446 | + |
| 447 | + const updatedCR = await prisma.change_Request.findUnique({ |
| 448 | + where: { crId: changeRequestId } |
| 449 | + }); |
| 450 | + |
| 451 | + expect(updatedCR?.reviewerId).toBe(leadershipUser2.userId); |
| 452 | + expect(updatedCR?.accepted).toBe(false); |
| 453 | + }); |
| 454 | + |
| 455 | + it('rejects member user from being requested as a reviewer', async () => { |
| 456 | + // requestCRReview should fail when trying to add a non-leadership user |
| 457 | + await expect( |
| 458 | + ChangeRequestsService.requestCRReview( |
| 459 | + submitterUser, |
| 460 | + [leadershipUser1.userId, memberUser.userId], |
| 461 | + changeRequestId, |
| 462 | + organization |
| 463 | + ) |
| 464 | + ).rejects.toThrow(AccessDeniedException); |
| 465 | + |
| 466 | + await expect( |
| 467 | + ChangeRequestsService.requestCRReview( |
| 468 | + submitterUser, |
| 469 | + [leadershipUser1.userId, memberUser.userId], |
| 470 | + changeRequestId, |
| 471 | + organization |
| 472 | + ) |
| 473 | + ).rejects.toThrow('The following user(s) are not leadership: Dick Grayson'); |
| 474 | + }); |
| 475 | + |
| 476 | + it('allows rejection by non-requested leadership when reviewers are requested', async () => { |
| 477 | + await ChangeRequestsService.requestCRReview(submitterUser, [leadershipUser1.userId], changeRequestId, organization); |
| 478 | + |
| 479 | + await expect( |
| 480 | + ChangeRequestsService.reviewChangeRequest( |
| 481 | + nonRequestedLeadership, |
| 482 | + changeRequestId, |
| 483 | + 'Rejecting this', |
| 484 | + false, |
| 485 | + organization, |
| 486 | + null |
| 487 | + ) |
| 488 | + ).rejects.toThrow(AccessDeniedException); |
| 489 | + }); |
| 490 | + }); |
287 | 491 | }); |
0 commit comments