Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(plugin-cc): add getQueues() method #4176

Open
wants to merge 2 commits into
base: wxcc
Choose a base branch
from

Conversation

rarajes2
Copy link
Contributor

@rarajes2 rarajes2 commented Mar 28, 2025

COMPLETES #< SPARK-567681 >

This pull request addresses

While performing consult or transfer we have option to choose the type of consult such as agent, DN, queue etc.
We needed to fetch the list of queues to perform the consult/transfer operation for queue as a result and for that we have added a method inside cc.ts file as getQueues().

by making the following changes

Added a method getQueues(orgId) inside the packages/@webex/plugin-cc/src/services/config/index.ts and exposed to the developer through cc.ts file with the same name.

We are able to get the list of queue.

Vidcast - https://app.vidcast.io/share/abe60753-715a-4811-b655-e0a40cb7c303

Change Type

  • Bug fix (non-breaking change which fixes an issue)
  • New feature (non-breaking change which adds functionality)
  • Breaking change (fix or feature that would cause existing functionality to change)
  • Documentation update
  • Tooling change
  • Internal code refactor

The following scenarios were tested

  • Able to get the list of queue
  • Filtering those queue inside the samples app to perform consult and transfer through queue.
    • Consult API is returning 202 status code but the next line is not being executed causing the buttons to remain in the same state --- NEED FURTHER TESTING AND CLARIFICATION
    • In case of error --- We are not able to get the server error message because we are creating our own error message and throwing that for the application to consume. NEED LARGER TEAM DISCUSSION ON THIS.
    • In case of Transfer it seems to be working fine.

Note: Based on the requirement mentioned on the ticket, the getQueues() method is working as expected.

I certified that

  • I have read and followed contributing guidelines

  • I discussed changes with code owners prior to submitting this pull request

  • I have not skipped any automated checks

  • All existing and new tests passed

  • I have updated the documentation accordingly


Make sure to have followed the contributing guidelines before submitting.

@rarajes2 rarajes2 requested a review from a team as a code owner March 28, 2025 13:58
Copy link
Contributor

coderabbitai bot commented Mar 28, 2025

Important

Review skipped

Auto reviews are disabled on base/target branches other than the default branch.

Please check the settings in the CodeRabbit UI or the .coderabbit.yaml file in this repository. To trigger a single review, invoke the @coderabbitai review command.

You can disable this status message by setting the reviews.review_status to false in the CodeRabbit configuration file.

📝 Walkthrough

Walkthrough

This pull request introduces a new asynchronous function in the contact center’s application code that retrieves a list of telephony queues via the webex.cc.getQueues() method. This function is integrated into the existing consult and transfer selection workflows (in the onConsultTypeSelectionChanged and onTransferTypeSelectionChanged functions) to populate dropdown options when the destination type is set to "queue". Additionally, the HTML has been modified by adjusting the default selections and updating placeholder text. In the SDK, a new public method getQueues has been added to the ContactCenter class which fetches queues using an updated endpoint defined in the configuration constants. The AgentConfigService is also enhanced with a new method to retrieve queue information, and corresponding TypeScript types (CallDistributionGroup and ContactServiceQueue) have been introduced. Finally, new unit tests have been added to validate the behavior of the getQueues functionality under different scenarios.

Possibly related PRs


🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai plan to trigger planning for file edits and PR creation.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

@rarajes2 rarajes2 added the validated If the pull request is validated for automation. label Mar 28, 2025
Copy link

This pull request is automatically being deployed by Amplify Hosting (learn more).

Access this pull request here: https://pr-4176.d3m3l2kee0btzx.amplifyapp.com

Copy link
Contributor

@Kesari3008 Kesari3008 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please update the description with the vidcast of testing done. Also update the section of 'by making the following changes'. We should mention there that what changes we did in the code to address the JIRA scope attached with the PR

@@ -487,4 +487,10 @@ export default class ContactCenter extends WebexPlugin implements IContactCenter
throw detailedError;
}
}

public async getQueues() {
const orgId = this.$webex.credentials.getOrgId();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What happens if we fail to fetch orgId here ?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added a check and throwing an error if not found.

@@ -487,4 +487,10 @@ export default class ContactCenter extends WebexPlugin implements IContactCenter
throw detailedError;
}
}

public async getQueues() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add the description for this public method

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

@@ -225,6 +225,18 @@ function toggleTransferOptions() {
transferOptionsElm.style.display = isTransferOptionsShown ? 'block' : 'none';
}

async function getQueueListForTelephonyChannel() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where are we checking for consult to queue feature flag to idetify if the queue should be show in the dropdown for consult?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Couldn't find the feature flag API. Not sure how we can add that check. Please let me know if you have more info on that, I will add that check here. For now it will throw error on the samples page if it's not enabled from control hub.


LoggerProxy.log('getQueues api success.', {module: CONFIG_FILE_NAME, method: 'getQueues'});

return Promise.resolve(response.body?.data);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just a thought: If body is undefined here then promise.resolve will happen with undefined data. Shouldn't we ensure that body is present before resolving the promise or if no queues are present. empty array is returned instead of undefined

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

body can't be undefined if the status code is 200. I had added the check just for any rare unknown case so that it does not break the application. We are not adding this check in any other methods. Please let me know if any change is required here.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also why do we need to promise resolve? Just returning the data is sufficient right? It is an async method anyway

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@mkesavan13 True, all other methods are doing the same. Kept it consistent with them.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it's okay to break the uniformity here or for future methods like this.

@@ -1003,4 +1013,26 @@ describe('webex.cc', () => {
expect(getErrorDetailsSpy).toHaveBeenCalledWith(error, 'startOutdial', CC_FILE);
});
});

describe('getQueues', () => {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What about failed scenario testcase?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added


const result = await webex.cc.getQueues();

expect(webex.cc.services.config.getQueues).toHaveBeenCalled();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please use toHaveBeenCalledWith and mention the orgId sent as a parameter

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

@@ -487,4 +487,10 @@ export default class ContactCenter extends WebexPlugin implements IContactCenter
throw detailedError;
}
}

public async getQueues() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is the public method from the SDK that external developers will use. Shouldn't this method have a return type mentioned.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done


const result = await agentConfigService.getQueues(mockOrgId);

expect(mockHttpRequest.request).toHaveBeenCalledWith({
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This expect statement should be present in failure testcases below too

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

@Kesari3008
Copy link
Contributor

@coderabbitai review

Copy link
Contributor

coderabbitai bot commented Apr 2, 2025

✅ Actions performed

Review triggered.

Note: CodeRabbit is an incremental review system and does not re-review already reviewed commits. This command is applicable only when automatic reviews are paused.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

♻️ Duplicate comments (2)
packages/@webex/plugin-cc/src/cc.ts (1)

491-495: 🛠️ Refactor suggestion

Add JSDoc and return type to public method.

As a public method in the SDK, this function should include JSDoc documentation and a return type annotation. It's also missing error handling that's present in other public methods.

Consider implementing like this:

- public async getQueues() {
+ /**
+  * Retrieves a list of queues for the organization.
+  * This is used for consult or transfer operations.
+  * 
+  * @returns {Promise<Array<ContactServiceQueue>>} A promise that resolves to a list of queues.
+  * @throws Error if retrieving queues fails
+  */
+ public async getQueues(): Promise<Array<ContactServiceQueue>> {
+   try {
     const orgId = this.$webex.credentials.getOrgId();
+     if (!orgId) {
+       throw new Error('Failed to retrieve organization ID');
+     }
     
     return this.services.config.getQueues(orgId);
+   } catch (error) {
+     const {error: detailedError} = getErrorDetails(error, 'getQueues', CC_FILE);
+     throw detailedError;
+   }
 }
packages/@webex/plugin-cc/test/unit/spec/cc.ts (1)

1017-1037: 🛠️ Refactor suggestion

Add test for failure scenario.

The test suite only covers the success case for the getQueues method. Add a test case for error handling to match the pattern of other test suites in this file.

Also, use toHaveBeenCalledWith to verify the orgId parameter is being passed correctly:

expect(webex.cc.services.config.getQueues).toHaveBeenCalled();
+ expect(webex.cc.services.config.getQueues).toHaveBeenCalledWith('mockOrgId');

Consider adding a failure test case:

it('should handle error during getQueues', async () => {
  const error = {
    details: {
      trackingId: '1234',
      data: {
        reason: 'Error retrieving queues',
      },
    },
  };
  
  webex.cc.services.config.getQueues = jest.fn().mockRejectedValue(error);
  
  await expect(webex.cc.getQueues()).rejects.toThrow(error.details.data.reason);
  
  expect(LoggerProxy.error).toHaveBeenCalledWith(
    `getQueues failed with trackingId: ${error.details.trackingId}`,
    {module: CC_FILE, method: 'getQueues'}
  );
  expect(getErrorDetailsSpy).toHaveBeenCalledWith(error, 'getQueues', CC_FILE);
});
🧹 Nitpick comments (1)
docs/samples/contact-center/index.html (1)

183-184: Capitalization changed in placeholder text.

The placeholder text has been capitalized from "Enter destination" to "Enter Destination". Consider maintaining consistent capitalization with the transfer input field on line 198 which still uses "Enter destination".

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 49c76aa and a3f710a.

📒 Files selected for processing (8)
  • docs/samples/contact-center/app.js (3 hunks)
  • docs/samples/contact-center/index.html (2 hunks)
  • packages/@webex/plugin-cc/src/cc.ts (1 hunks)
  • packages/@webex/plugin-cc/src/services/config/constants.ts (1 hunks)
  • packages/@webex/plugin-cc/src/services/config/index.ts (2 hunks)
  • packages/@webex/plugin-cc/src/services/config/types.ts (1 hunks)
  • packages/@webex/plugin-cc/test/unit/spec/cc.ts (4 hunks)
  • packages/@webex/plugin-cc/test/unit/spec/services/config/index.ts (1 hunks)
🧰 Additional context used
🧬 Code Definitions (3)
packages/@webex/plugin-cc/src/services/config/index.ts (5)
packages/@webex/plugin-cc/src/services/config/types.ts (1)
  • ContactServiceQueue (594-624)
packages/@webex/plugin-cc/src/services/config/constants.ts (1)
  • endPointMap (16-47)
packages/@webex/plugin-cc/src/services/constants.ts (1)
  • WCC_API_GATEWAY (2-2)
packages/@webex/plugin-cc/src/types.ts (2)
  • HTTP_METHODS (9-15)
  • HTTP_METHODS (18-18)
packages/@webex/plugin-cc/src/constants.ts (1)
  • CONFIG_FILE_NAME (8-8)
packages/@webex/plugin-cc/test/unit/spec/services/config/index.ts (2)
packages/@webex/plugin-cc/src/logger-proxy.ts (1)
  • LoggerProxy (4-48)
packages/@webex/plugin-cc/src/constants.ts (1)
  • CONFIG_FILE_NAME (8-8)
docs/samples/contact-center/app.js (2)
packages/@webex/test-helper-mock-webex/src/index.js (1)
  • webex (222-222)
packages/@webex/plugin-cc/src/logger-proxy.ts (1)
  • error (35-39)
🔇 Additional comments (15)
docs/samples/contact-center/index.html (2)

177-180: Default dropdown selection now set to "Dial Number".

The default selection for the consult destination type has been changed from "Queue" to "Dial Number", which aligns with the changes made to the transfer options below.


193-196: Default dropdown selection now set to "Dial Number".

The default selection for the transfer destination type has been changed from "Queue" to "Dial Number".

packages/@webex/plugin-cc/src/services/config/constants.ts (1)

46-46: Added new endpoint mapping for queue list.

The new endpoint mapping for retrieving the list of queues follows the existing pattern and facilitates the new functionality for displaying available queues.

docs/samples/contact-center/app.js (4)

228-237: LGTM: Good implementation of the queue retrieval function.

The implementation of getQueueListForTelephonyChannel is well-structured with proper error handling through a try-catch block and appropriate logging of failures.


228-228: Check for feature flag for the consult to queue capability.

Where are we checking for consult to queue feature flag to identify if the queue should be shown in the dropdown for consult? This was previously noted in a review comment.

#!/bin/bash
# Search for feature flag checks related to consultToQueue
rg -A 2 "consultToQueue" --glob "*.ts" --glob "*.js" --glob "*.json"

261-275: Properly handles queue selection in consult options.

The implementation correctly fetches the queue list and creates a dropdown with options based on queue names and IDs when the destination type is set to 'queue'.


304-317: Properly handles queue selection in transfer options.

The implementation correctly fetches the queue list and creates a dropdown with options based on queue names and IDs when the transfer destination type is set to 'queue'.

packages/@webex/plugin-cc/src/services/config/index.ts (3)

536-564: Consider handling undefined response data.

When Promise.resolve is called with response.body?.data, there's no fallback if body or data is undefined. The method is typed to return Promise<ContactServiceQueue[]>, but could potentially return Promise<undefined>.

Consider modifying line 556 to return an empty array instead of undefined in such cases:

-      return Promise.resolve(response.body?.data);
+      return Promise.resolve(response.body?.data || []);

536-540: Good method documentation.

The method has appropriate JSDoc comments explaining its purpose, parameters, and return type.


541-564: LGTM: Implementation follows established patterns.

The implementation follows the same pattern as other API methods in the class with proper error handling, logging, and response status checking.

packages/@webex/plugin-cc/test/unit/spec/services/config/index.ts (3)

931-959: Good test coverage for successful API response.

The test correctly verifies that the method makes the appropriate API call and processes the response as expected.


961-969: Complete error testing.

The test properly verifies error handling when the API call fails.


971-982: Complete non-200 response testing.

The test properly verifies error handling when the API returns a non-200 status code.

packages/@webex/plugin-cc/src/services/config/types.ts (2)

588-592: Well-defined CallDistributionGroup type.

The type definition for CallDistributionGroup includes all necessary properties with appropriate types.


594-624: Comprehensive ContactServiceQueue type definition.

The ContactServiceQueue type is well-structured with a comprehensive list of properties that cover all aspects of a queue configuration.

Copy link
Contributor

@mkesavan13 mkesavan13 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just a few nitpick comments


LoggerProxy.log('getQueues api success.', {module: CONFIG_FILE_NAME, method: 'getQueues'});

return Promise.resolve(response.body?.data);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also why do we need to promise resolve? Just returning the data is sufficient right? It is an async method anyway

Copy link
Contributor

@mkesavan13 mkesavan13 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good to me mostly. Just one nitpick about the promise resolution step. I've resolved the rest.

Thanks for the PR, @rarajes2

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
validated If the pull request is validated for automation.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants