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

chore(common): apply decorators function #3999

Open
wants to merge 1 commit into
base: next
Choose a base branch
from

Conversation

Coread
Copy link
Collaborator

@Coread Coread commented Nov 21, 2024

COMPLETES #< INSERT LINK TO ISSUE >

This pull request addresses

Decorators cause lots of errors in VScode due to the way they are trying to apply to ampersand state objects.

by making the following changes

I've created a new function to apply the decorators manually. This avoids the need to physically decorate the code and so the linting works again.

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

< ENUMERATE TESTS PERFORMED, WHETHER MANUAL OR AUTOMATED >

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.

Summary by CodeRabbit

  • New Features

    • Introduced the applyDecorators function for applying decorators to methods, enhancing functionality.
    • Enhanced the Device class with centralized decorator application for improved readability.
  • Bug Fixes

    • Updated method signatures for refresh and register for clearer parameter naming.
  • Tests

    • Added a test suite for the applyDecorators function to ensure correct functionality.

@Coread Coread requested review from a team as code owners November 21, 2024 11:15
Copy link

coderabbitai bot commented Nov 21, 2024

Walkthrough

The changes introduce a new function applyDecorators in apply-decorators.js, which applies multiple decorators to specified methods of an object. The function is then exported for use in other modules. Additionally, the index.js file is updated to include this new export. A test suite for applyDecorators is added to ensure its functionality. Furthermore, the Device class in device.js is modified to utilize applyDecorators for several methods, enhancing the structure and clarity of the code.

Changes

File Path Change Summary
packages/@webex/common/src/apply-decorators.js Added method applyDecorators(object, decoratedMethods) and default export { applyDecorators }.
packages/@webex/common/src/index.js Added export statement for applyDecorators: export {default as applyDecorators} from './apply-decorators';
packages/@webex/common/test/unit/spec/apply-decorators.js Introduced a test suite for applyDecorators to verify correct decorator application.
packages/@webex/internal-plugin-device/src/device.js Updated Device methods to use applyDecorators, changed parameter names for clarity, marked markUrlFailedAndGetNew as deprecated.

Sequence Diagram(s)

sequenceDiagram
    participant User
    participant Device
    participant Decorators

    User->>Device: Calls method (e.g., register)
    Device->>Decorators: Applies decorators
    Decorators->>Device: Modifies method behavior
    Device->>User: Returns result
Loading

🐰 "In the code where decorators play,
A rabbit hops and leads the way.
With methods wrapped in a fancy dress,
Our functions shine, we must confess!
So let’s rejoice, in code we trust,
For every change, we find a new must!" 🐇

Warning

There were issues while running some tools. Please review the errors and either fix the tool’s configuration or disable the tool if it’s a critical failure.

🔧 eslint

If the error stems from missing dependencies, add them to the package.json file. For unrecoverable errors (e.g., due to private dependencies), disable the tool in the CodeRabbit configuration.

warning eslint@8.57.1: This version is no longer supported. Please see https://eslint.org/version-support for other options.
warning eslint > @humanwhocodes/config-array@0.13.0: Use @eslint/config-array instead
warning eslint > @humanwhocodes/config-array > @humanwhocodes/object-schema@2.0.3: Use @eslint/object-schema instead
warning eslint > file-entry-cache > flat-cache > rimraf@3.0.2: Rimraf versions prior to v4 are no longer supported
warning eslint > file-entry-cache > flat-cache > rimraf > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning eslint > file-entry-cache > flat-cache > rimraf > glob > inflight@1.0.6: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.
warning eslint-import-resolver-typescript > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning glob@7.2.3: Glob versions prior to v9 are no longer supported
warning intern > glob@7.1.7: Glob versions prior to v9 are no longer supported
warning intern > glob > inflight@1.0.6: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.
warning jasmine > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning jest > @jest/core > jest-config > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning jest > @jest/core > jest-runtime > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning jest > @jest/core > @jest/reporters > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning jest > @jest/core > @jest/transform > babel-plugin-istanbul > test-exclude > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning mocha > glob@7.2.0: Glob versions prior to v9 are no longer supported
warning mocha > glob > inflight@1.0.6: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.
warning sinon@9.2.4: 16.1.1
warning wd > archiver > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning wd > q@1.5.1: You or someone you depend on is using Q, the JavaScript Promise library that gave JavaScript developers strong feelings about promises. They can almost certainly migrate to the native JavaScript promise now. Thank you literally everyone for joining me in this bet against the odds. Be excellent to each other.

(For a CapTP with native promises, see @endo/eventual-send and @endo/captp)
warning wd > request@2.88.0: request has been deprecated, see request/request#3142
warning wd > archiver > archiver-utils > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning wd > request > uuid@3.4.0: Please upgrade to version 7 or higher. Older versions may use Math.random() in certain circumstances, which is known to be problematic. See https://v8.dev/blog/math-random for details.
warning wd > request > har-validator@5.1.5: this library is no longer supported
warning @babel/cli > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning @babel/plugin-proposal-async-generator-functions@7.20.7: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-async-generator-functions instead.
warning @babel/plugin-proposal-class-properties@7.18.6: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-class-properties instead.
warning @babel/plugin-proposal-export-namespace-from@7.18.9: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-export-namespace-from instead.
warning @babel/plugin-proposal-nullish-coalescing-operator@7.18.6: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-nullish-coalescing-operator instead.
warning @babel/plugin-proposal-object-rest-spread@7.20.7: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-object-rest-spread instead.
warning @babel/plugin-proposal-optional-chaining@7.21.0: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-optional-chaining instead.
warning @babel/polyfill@7.12.1: 🚨 This package has been deprecated in favor of separate inclusion of a polyfill and regenerator-runtime (when needed). See the @babel/polyfill docs (https://babeljs.io/docs/en/babel-polyfill) for more information.
warning @babel/polyfill > core-js@2.6.12: core-js@<3.23.3 is no longer maintained and not recommended for usage due to the number of issues. Because of the V8 engine whims, feature detection in old core-js versions could cause a slowdown up to 100x even if nothing is polyfilled. Some versions have web compatibility issues. Please, upgrade your dependencies to the actual version of core-js.
warning @babel/runtime-corejs2 > core-js@2.6.12: core-js@<3.23.3 is no longer maintained and not recommended for usage due to the number of issues. Because of the V8 engine whims, feature detection in old core-js versions could cause a slowdown up to 100x even if nothing is polyfilled. Some versions have web compatibility issues. Please, upgrade your dependencies to the actual version of core-js.
warning babel-plugin-lodash > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > eslint@8.57.1: This version is no longer supported. Please see https://eslint.org/version-support for other options.
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > glob@7.2.3: Glob versions prior to v9 are no longer supported
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > sinon@9.2.4: 16.1.1
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > @babel/plugin-proposal-async-generator-functions@7.20.7: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-async-generator-functions instead.
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > @babel/plugin-proposal-class-properties@7.18.6: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-class-properties instead.
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > @babel/plugin-proposal-export-namespace-from@7.18.9: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-export-namespace-from instead.
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > @babel/plugin-proposal-nullish-coalescing-operator@7.18.6: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-nullish-coalescing-operator instead.
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > @babel/plugin-proposal-object-rest-spread@7.20.7: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-object-rest-spread instead.
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > @babel/plugin-proposal-optional-chaining@7.21.0: This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-optional-chaining instead.
warning workspace-aggregator-bdf8de00-66ba-4235-be5d-dc219ca74a9a > @babel/polyfill@7.12.1: 🚨 This package has been deprecated in favor of separate inclusion of a polyfill and regenerator-runtime (when needed). See the @babel/polyfill docs (https://babeljs.io/docs/en/babel-polyfill) for more information.
error eslint-plugin-jsdoc@40.3.0: The engine "node" is incompatible with this module. Expected version "^14 || ^16 || ^17 || ^18 || ^19". Got "22.9.0"
error Found incompatible module.


Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 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 resolve resolve all the CodeRabbit review comments.
  • @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.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

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.

Copy link

@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: 3

🧹 Outside diff range and nitpick comments (3)
packages/@webex/common/src/apply-decorators.js (1)

1-8: Improve JSDoc documentation accuracy and completeness.

The JSDoc comments should be enhanced with more specific type information and a better description of the decoratedMethods parameter structure.

 /**
  * This applies decorators to an ampersand state object
  * It wraps the function in the order the decorators are provided
  * This means that when you call the method, the decorators are applied in reverse order
- * @param {*} object - The object to apply the decorators to
- * @param {*} decoratedMethods
- * @returns {undefined}
+ * @param {Object} object - The object to apply the decorators to
+ * @param {Object.<string, Function[]>} decoratedMethods - Map of method names to their decorator functions
  */
packages/@webex/common/test/unit/spec/apply-decorators.js (1)

25-33: Document decorator application order

While the test correctly verifies the decorator order, it would be helpful to add a comment explaining that decorators are applied in reverse order (last decorator wraps first). This is a common source of confusion in decorator implementations.

 applyDecorators(obj, {
+  // Decorators are applied in reverse order:
+  // decorator2 wraps decorator1, which wraps the original method
   method: [decorator1, decorator2]
 });
packages/@webex/internal-plugin-device/src/device.js (1)

370-370: Great improvement in parameter naming!

The change from options to deviceRegistrationOptions enhances code clarity by making the expected type more explicit while maintaining backward compatibility.

Consider adding TypeScript-style JSDoc type annotations to further improve type safety:

- @param {Object} deviceRegistrationOptions - The options for registration.
+ @param {Object} deviceRegistrationOptions - The options for registration.
+ @typedef {Object} DeviceRegistrationOptions
+ @property {CatalogDetails} [includeDetails] - The details to include in the refresh/register request.

Also applies to: 446-446

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 8d6d1bc and f16e092.

📒 Files selected for processing (4)
  • packages/@webex/common/src/apply-decorators.js (1 hunks)
  • packages/@webex/common/src/index.js (1 hunks)
  • packages/@webex/common/test/unit/spec/apply-decorators.js (1 hunks)
  • packages/@webex/internal-plugin-device/src/device.js (4 hunks)
🔇 Additional comments (7)
packages/@webex/common/src/apply-decorators.js (2)

1-20: Verify that this resolves the VSCode linting issues.

Since this PR aims to fix VSCode linting issues with decorators on ampersand state objects, please confirm:

  1. Have you tested this with VSCode to ensure the linting errors are resolved?
  2. Are there any specific VSCode settings or extensions that need to be documented?
✅ Verification successful

The implementation should resolve VSCode linting issues

Based on the codebase analysis:

  1. The repository has TypeScript configuration with experimentalDecorators enabled, which is the correct setting for decorator support.
  2. The implementation moves away from using decorators directly on classes to using a programmatic approach with applyDecorators, which is a common pattern to avoid linting issues.
  3. The change is consistent with the codebase's approach to handling decorators in other files.
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check for any remaining decorator usage that might still cause linting issues
rg -t js "@\w+\s*\n*\s*class"
rg -t js "@\w+\s*\n*\s*\w+\s*\([^)]*\)\s*{"

Length of output: 397


Script:

#!/bin/bash
# Let's try a different approach to find decorator usage
# Search for @ symbol followed by word characters
rg -U "@\w+" --type js

# Also check for any configuration files that might affect VSCode linting
fd -e json -e js "\.eslint" -X cat {}

Length of output: 130818


Script:

#!/bin/bash
# Let's check for any ESLint configuration files that might affect decorator usage
fd -e json -e js -e yaml -e yml "eslint|\.eslintrc" -X cat {}

# Also check for any VSCode settings that might be relevant
fd -g "settings.json" -X cat {}

# And check for any TypeScript configuration that might affect decorators
fd -g "tsconfig*.json" -X cat {}

Length of output: 10131


9-16: Consider preserving property descriptors.

The current implementation might not properly handle property descriptors, which could affect how decorators behave with getters/setters or non-configurable properties.

Let's check if there are any getters/setters in the codebase that might be decorated:

packages/@webex/common/test/unit/spec/apply-decorators.js (2)

7-23: Well-structured test setup with proper function binding

The test setup is well-designed:

  • Uses sinon stub for method tracking
  • Correctly implements decorators with proper 'this' binding
  • Clear demonstration of decorator behavior

1-3: Verify the new utility module's structure

The import path suggests this is a new shared utility. Let's verify its structure and exports.

✅ Verification successful

Import structure and module location are correct

The utility module exists at the expected location with proper exports. The implementation shows a well-documented utility function for applying decorators to ampersand state objects, and it's currently only imported by its test file. The import structure in the test file is consistent with the module's exports and follows the project's conventions.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the new utility module structure and exports

# Check if the module exists and exports the function
rg -g "src/apply-decorators.js" -A 5 "export.*applyDecorators"

# Check if there are any other files importing this utility
rg "from ['\"]\@webex\/common\/src\/apply-decorators['\"]"

Length of output: 403


Script:

#!/bin/bash
# Let's try a different approach to verify the module structure

# Find the apply-decorators.js file
fd "apply-decorators.js" 

# Check the content of the file if found
fd "apply-decorators.js" --exec cat {}

# Search for imports using ripgrep with proper escaping
rg "from ['\"]@webex/common/src/apply-decorators['\"]"

Length of output: 1789

packages/@webex/common/src/index.js (1)

24-24: LGTM! The export is well-placed and follows consistent patterns.

The new export for applyDecorators is correctly positioned among related utilities and maintains the file's organization.

Let's verify the implementation of the applyDecorators utility:

✅ Verification successful

Implementation and usage of applyDecorators looks solid and serves its purpose

The implementation in apply-decorators.js is well-designed:

  • Takes an object and a map of method-to-decorators
  • Correctly applies multiple decorators in the expected order (reverse execution)
  • Has proper JSDoc documentation

Current usage shows:

  • Used in @webex/internal-plugin-device/src/device.js to apply decorators programmatically
  • Has unit tests in @webex/common/test/unit/spec/apply-decorators.js
  • The codebase shows extensive use of decorators that could benefit from this utility
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the implementation and usage of applyDecorators

# Check the implementation
echo "Checking applyDecorators implementation:"
cat packages/@webex/common/src/apply-decorators.js

# Find usages to understand the impact
echo -e "\nChecking usages across the codebase:"
rg "applyDecorators" --type js

# Look for any existing decorator-related issues
echo -e "\nChecking for existing decorator patterns that might need migration:"
rg -B2 -A2 "@\w+\s*\([^)]*\)\s*$" --type js

Length of output: 12983

packages/@webex/internal-plugin-device/src/device.js (2)

2-2: LGTM! Import changes align with PR objectives

The addition of applyDecorators import supports the goal of manually applying decorators to resolve VSCode linting issues.


886-894: Well-structured decorator application!

The centralized application of decorators using applyDecorators:

  1. Resolves VSCode linting issues as intended
  2. Improves code maintainability by grouping all decorators in one place
  3. Makes decorator application more explicit and easier to manage

Let's verify that all decorated methods are properly exported and accessible:

✅ Verification successful

Decorator application correctly matches implemented methods

The verification confirms that all methods decorated in the applyDecorators call are properly implemented in the Device class:

  • register(deviceRegistrationOptions = {})
  • refresh(deviceRegistrationOptions = {})
  • unregister()
  • markUrlFailedAndGetNew(url)
  • initialize(...args)
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Verify that all decorated methods exist in the Device class
ast-grep --pattern 'class Device {
  $$$
  register($$$) {
    $$$
  }
  refresh($$$) {
    $$$
  }
  unregister($$$) {
    $$$
  }
  markUrlFailedAndGetNew($$$) {
    $$$
  }
  initialize($$$) {
    $$$
  }
  $$$
}'

Length of output: 218


Script:

#!/bin/bash
# Let's try a different approach to verify the methods individually
echo "=== Checking register method ==="
ast-grep --pattern 'register($$$) {
  $$$
}'

echo "=== Checking refresh method ==="
ast-grep --pattern 'refresh($$$) {
  $$$
}'

echo "=== Checking unregister method ==="
ast-grep --pattern 'unregister($$$) {
  $$$
}'

echo "=== Checking markUrlFailedAndGetNew method ==="
ast-grep --pattern 'markUrlFailedAndGetNew($$$) {
  $$$
}'

echo "=== Checking initialize method ==="
ast-grep --pattern 'initialize($$$) {
  $$$
}'

# Also let's check the Device class definition
echo "=== Checking Device class ==="
ast-grep --pattern 'class Device extends $$$'

Length of output: 777


Script:

#!/bin/bash
# Let's try using ripgrep to search for these methods and class
echo "=== Checking Device class and methods ==="
rg -A 2 "class Device" packages/@webex/internal-plugin-device/src/device.js

echo "=== Checking method implementations ==="
rg "(register|refresh|unregister|markUrlFailedAndGetNew|initialize)\s*\([^)]*\)\s*{" packages/@webex/internal-plugin-device/src/device.js -A 2

Length of output: 626


Script:

#!/bin/bash
# Let's try a simpler ripgrep approach
echo "=== Checking Device class ==="
rg "class Device" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "=== Checking individual methods ==="
echo "--- register ---"
rg "register.*{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- refresh ---"
rg "refresh.*{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- unregister ---"
rg "unregister.*{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- markUrlFailedAndGetNew ---"
rg "markUrlFailedAndGetNew.*{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- initialize ---"
rg "initialize.*{" packages/@webex/internal-plugin-device/src/device.js -A 2

Length of output: 1342


Script:

#!/bin/bash
# Let's try with escaped curly braces in the regex
echo "=== Checking Device class ==="
rg "class Device" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "=== Checking individual methods ==="
echo "--- register ---"
rg "register.*\{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- refresh ---"
rg "refresh.*\{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- unregister ---"
rg "unregister.*\{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- markUrlFailedAndGetNew ---"
rg "markUrlFailedAndGetNew.*\{" packages/@webex/internal-plugin-device/src/device.js -A 2

echo "--- initialize ---"
rg "initialize.*\{" packages/@webex/internal-plugin-device/src/device.js -A 2

Length of output: 2010

Comment on lines +18 to +20
export default {
applyDecorators,
};
Copy link

Choose a reason for hiding this comment

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

💡 Codebase verification

Consider removing default export since only named export is used

The codebase exclusively uses the named import {applyDecorators}, making the default export unnecessary. For better code maintainability and to avoid confusion, consider removing the default export and keeping only the named export.

  • packages/@webex/common/src/apply-decorators.js: Remove default export and keep only export { applyDecorators }
🔗 Analysis chain

Verify import usage consistency across the codebase.

Having both named and default exports is fine, but let's ensure they're used consistently.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check how this module is imported across the codebase
rg -t js "import.*from.*apply-decorators" 

Length of output: 169

Comment on lines +9 to +16
export function applyDecorators(object, decoratedMethods) {
Object.entries(decoratedMethods).forEach(([method, decorators]) => {
object[method] = decorators.reduce(
(decorated, decorator) => decorator(decorated),
object[method]
);
});
}
Copy link

Choose a reason for hiding this comment

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

⚠️ Potential issue

Add input validation to prevent runtime errors.

The function should validate its inputs to ensure robust operation and provide clear error messages.

 export function applyDecorators(object, decoratedMethods) {
+  if (!object || typeof object !== 'object') {
+    throw new TypeError('First argument must be an object');
+  }
+  if (!decoratedMethods || typeof decoratedMethods !== 'object') {
+    throw new TypeError('Second argument must be an object mapping methods to decorators');
+  }
+
   Object.entries(decoratedMethods).forEach(([method, decorators]) => {
+    if (!Array.isArray(decorators)) {
+      throw new TypeError(`Decorators for method "${method}" must be an array`);
+    }
+    if (typeof object[method] !== 'function') {
+      throw new TypeError(`"${method}" is not a method on the target object`);
+    }
+    if (!decorators.every((d) => typeof d === 'function')) {
+      throw new TypeError(`All decorators for method "${method}" must be functions`);
+    }
+
     object[method] = decorators.reduce(
       (decorated, decorator) => decorator(decorated),
       object[method]
     );
   });
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
export function applyDecorators(object, decoratedMethods) {
Object.entries(decoratedMethods).forEach(([method, decorators]) => {
object[method] = decorators.reduce(
(decorated, decorator) => decorator(decorated),
object[method]
);
});
}
export function applyDecorators(object, decoratedMethods) {
if (!object || typeof object !== 'object') {
throw new TypeError('First argument must be an object');
}
if (!decoratedMethods || typeof decoratedMethods !== 'object') {
throw new TypeError('Second argument must be an object mapping methods to decorators');
}
Object.entries(decoratedMethods).forEach(([method, decorators]) => {
if (!Array.isArray(decorators)) {
throw new TypeError(`Decorators for method "${method}" must be an array`);
}
if (typeof object[method] !== 'function') {
throw new TypeError(`"${method}" is not a method on the target object`);
}
if (!decorators.every((d) => typeof d === 'function')) {
throw new TypeError(`All decorators for method "${method}" must be functions`);
}
object[method] = decorators.reduce(
(decorated, decorator) => decorator(decorated),
object[method]
);
});
}

Comment on lines +5 to +6
describe('applyDecorators()', () => {
it('applies the decorators in order', () => {
Copy link

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Consider adding more test cases

The test suite currently only verifies the order of decorator application. Given that this PR addresses VSCode issues with ampersand state objects, consider adding test cases for:

  • Error handling scenarios
  • Edge cases (e.g., empty decorator array, undefined methods)
  • Integration with ampersand state objects

Would you like me to help generate these additional test cases?

Copy link

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

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

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant