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

Relocate selenium-webdriver to root directory #15509

Merged
merged 4 commits into from
Mar 25, 2025

Conversation

harsha509
Copy link
Member

@harsha509 harsha509 commented Mar 25, 2025

User description

Thanks for contributing to Selenium!
A PR well described will help maintainers to quickly review and merge it

Before submitting your PR, please check our contributing guidelines.
Avoid large PRs, help reviewers by making them as simple and short as possible.

Motivation and Context

Types of changes

  • 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)

Checklist

  • I have read the contributing document.
  • My change requires a change to the documentation.
  • I have updated the documentation accordingly.
  • I have added tests to cover my changes.
  • All new and existing tests passed.

PR Type

Enhancement, Tests


Description

  • Relocated selenium-webdriver to the root directory for better project organization.

  • Enhanced WebDriver functionalities with new modules for Chromium and Firefox clients, including custom commands and options.

  • Added comprehensive test suites for WebDriver functionalities, input actions, BiDi protocol, DevTools integration, and virtual authenticator.

  • Introduced new utilities for file system operations, logging, and browsing context management.

  • Refactored the main module to improve WebDriver API and configuration handling.

  • Implemented network-related types and classes for BiDi protocol.

  • Deleted outdated files from the node directory to streamline the codebase.


Changes walkthrough 📝

Relevant files
Tests
6 files
webdriver_test.js
Comprehensive test suite for WebDriver functionalities.   

javascript/selenium-webdriver/test/lib/webdriver_test.js

  • Added a comprehensive test suite for WebDriver functionalities.
  • Introduced mock classes like FakeExecutor and Expectation for testing.
  • Implemented tests for session creation, element interactions, actions,
    and error handling.
  • Covered edge cases like invalid inputs, timeouts, and asynchronous
    behaviors.
  • [link]   
    script_test.js
    Added BiDi script manager tests for various scenarios.     

    javascript/selenium-webdriver/test/bidi/script_test.js

  • Added comprehensive tests for BiDi script manager functionality.
  • Included tests for function calls, sandboxing, and realm-specific
    operations.
  • Verified exception handling and ownership configurations in BiDi
    scripts.
  • Added tests for preload script management and event listeners.
  • [link]   
    input_test.js
    Added tests for input actions and device synchronization.

    javascript/selenium-webdriver/test/lib/input_test.js

  • Added tests for input actions including key, mouse, and drag-and-drop
    operations.
  • Verified synchronization and asynchronous behavior of input actions.
  • Included edge cases for invalid arguments and large input strings.
  • Tested device-specific and multi-device input scenarios.
  • [link]   
    devtools_test.js
    Added tests for DevTools integration and event handling. 

    javascript/selenium-webdriver/test/devtools_test.js

  • Added tests for DevTools integration with WebDriver.
  • Verified network interception, basic auth injection, and DOM mutation
    events.
  • Included tests for JavaScript console and exception event listeners.
  • Tested DevTools commands for page and network operations.
  • [link]   
    promise_test.js
    Added extensive tests for promise utilities in WebDriver.

    javascript/selenium-webdriver/test/lib/promise_test.js

  • Added comprehensive test cases for promise utilities in Selenium
    WebDriver.
  • Covered scenarios like promise resolution, rejection, and chaining.
  • Included tests for fullyResolved, finally, checkedNodeCall, map, and
    filter methods.
  • Ensured proper error handling and edge case coverage.
  • [link]   
    virtualAuthenticator_test.js
    Added tests for virtual authenticator functionality in WebDriver.

    javascript/selenium-webdriver/test/virtualAuthenticator_test.js

  • Added test cases for virtual authenticator functionality in Selenium
    WebDriver.
  • Covered scenarios like creating, removing, and managing credentials.
  • Included tests for resident and non-resident credentials with various
    protocols.
  • Ensured proper error handling and validation for unsupported
    operations.
  • [link]   
    Enhancement
    8 files
    input.js
    Input handling and user gesture simulation enhancements. 

    javascript/selenium-webdriver/lib/input.js

  • Added classes and enums for user input handling, including Key,
    Button, and Action.
  • Introduced Device subclasses like Keyboard, Pointer, and Wheel for
    input simulation.
  • Implemented the Actions class for complex user gesture sequences.
  • Provided utility functions for key validation and input device
    management.
  • [link]   
    index.js
    Refactored main module for WebDriver API and configurations.

    javascript/selenium-webdriver/index.js

  • Refactored the main module to export WebDriver's primary API.
  • Added support for browser-specific configurations via the Builder
    class.
  • Introduced utility functions for managing WebDriver sessions and
    capabilities.
  • Enhanced compatibility with environment variables for runtime
    configuration.
  • [link]   
    chromium.js
    Introduced Chromium WebDriver client with custom commands.

    javascript/selenium-webdriver/chromium.js

  • Added a new module for Chromium-based WebDriver clients.
  • Implemented classes for managing Chromium-specific options, services,
    and drivers.
  • Defined custom commands and executors for Chromium WebDriver.
  • Provided extensive documentation and examples for usage.
  • [link]   
    firefox.js
    Introduced Firefox WebDriver client with custom options. 

    javascript/selenium-webdriver/firefox.js

  • Added a new module for Firefox WebDriver client.
  • Implemented classes for managing Firefox-specific options, profiles,
    and services.
  • Defined custom commands and executors for Firefox WebDriver.
  • Included support for extensions, mobile features, and debugging.
  • [link]   
    networkTypes.js
    Introduced network-related types and classes for BiDi protocol.

    javascript/selenium-webdriver/bidi/networkTypes.js

  • Added a new file defining network-related types and classes for BiDi
    protocol.
  • Implemented classes like BytesValue, Header, Cookie, FetchTimingInfo,
    and RequestData for handling network data.
  • Introduced utility methods for serialization and data manipulation.
  • Added error handling for invalid arguments in constructors and
    methods.
  • [link]   
    browsingContext.js
    Added browsing context management for BiDi protocol.         

    javascript/selenium-webdriver/bidi/browsingContext.js

  • Added a new file defining BrowsingContext and related classes for BiDi
    protocol.
  • Implemented methods for navigation, screenshot capture, and context
    management.
  • Introduced Locator class for node location and NavigateResult for
    navigation results.
  • Included error handling for invalid arguments and unsupported
    operations.
  • [link]   
    logging.js
    Implemented a hierarchical logging system for WebDriver. 

    javascript/selenium-webdriver/lib/logging.js

  • Added a new file implementing a logging system for WebDriver.
  • Defined Level, Entry, Logger, and LogManager classes for hierarchical
    logging.
  • Provided methods for adding/removing handlers and logging messages at
    various levels.
  • Included utilities for console logging and log preferences management.
  • [link]   
    index.js
    Introduced file system utilities for WebDriver.                   

    javascript/selenium-webdriver/io/index.js

  • Added a new file providing file system utilities for Selenium
    WebDriver.
  • Implemented functions for directory management, file copying, and
    temporary file handling.
  • Included utilities for recursive operations like directory removal and
    walking.
  • Added error handling for invalid paths and unsupported operations.
  • [link]   
    Additional files
    101 files
    .bazelignore +1/-1     
    nightly.yml +2/-2     
    update-documentation.yml +1/-1     
    .skipped-tests +8/-8     
    BUILD.bazel +1/-1     
    CONTRIBUTING.md +2/-2     
    MODULE.bazel +1/-1     
    README.md +2/-2     
    Rakefile +14/-14 
    BUILD.bazel +1/-1     
    BUILD.bazel +1/-1     
    BUILD.bazel +1/-1     
    BUILD.bazel +1/-1     
    BUILD.bazel +3/-3     
    BUILD.bazel +1/-1     
    BUILD.bazel +2/-2     
    BUILD.bazel +1/-1     
    BUILD.bazel +1/-1     
    BUILD.bazel +2/-2     
    BUILD.bazel +2/-2     
    deploy.js +0/-163 
    global.js +0/-115 
    jszip.js +0/-67   
    mocha.js +0/-40   
    rimraf.js +0/-26   
    tmp.js +0/-54   
    ws.js +0/-49   
    xml2js.js +0/-31   
    gendocs.js +0/-274 
    optparse.js +0/-553 
    mocha_test.bzl +1/-1     
    .npmignore [link]   
    .prettierignore [link]   
    .prettierrc [link]   
    BUILD.bazel +7/-7     
    CHANGES.md +1/-1     
    README.md [link]   
    addInterceptParameters.js [link]   
    argumentValue.js [link]   
    browser.js [link]   
    browsingContextInspector.js [link]   
    browsingContextTypes.js [link]   
    captureScreenshotParameters.js [link]   
    clientWindowInfo.js [link]   
    clipRectangle.js [link]   
    continueRequestParameters.js [link]   
    continueResponseParameters.js [link]   
    cookieFilter.js [link]   
    createContextParameters.js [link]   
    evaluateResult.js [link]   
    permissions.js [link]   
    filterBy.js [link]   
    index.js [link]   
    input.js [link]   
    interceptPhase.js [link]   
    logEntries.js [link]   
    logInspector.js [link]   
    network.js [link]   
    networkInspector.js [link]   
    partialCookie.js [link]   
    partitionDescriptor.js [link]   
    partitionKey.js [link]   
    protocolType.js [link]   
    protocolValue.js [link]   
    provideResponseParameters.js [link]   
    realmInfo.js [link]   
    resultOwnership.js [link]   
    scriptManager.js [link]   
    scriptTypes.js [link]   
    storage.js [link]   
    urlPattern.js [link]   
    chrome.js [link]   
    driverFinder.js [link]   
    seleniumManager.js [link]   
    CDPConnection.js [link]   
    networkinterceptor.js [link]   
    edge.js [link]   
    eslint.config.js [link]   
    chrome_android.js [link]   
    chrome_mobile_emulation.js [link]   
    google_search.js [link]   
    google_search_test.js [link]   
    headless.js [link]   
    logging.js [link]   
    index.js [link]   
    util.js [link]   
    ie.js [link]   
    exec.js [link]   
    zip.js [link]   
    jsdoc_conf.json [link]   
    README [link]   
    BUILD.bazel +5/-5     
    make-atoms-module.js [link]   
    by.js [link]   
    capabilities.js [link]   
    command.js [link]   
    error.js [link]   
    account.js [link]   
    dialog.js [link]   
    http.js +3/-3     
    Additional files not shown

    Need help?
  • Type /help how to ... in the comments thread for any questions about Qodo Merge usage.
  • Check out the documentation for more information.
  • Copy link
    Contributor

    PR Reviewer Guide 🔍

    Here are some key observations to aid the review process:

    ⏱️ Estimated effort to review: 3 🔵🔵🔵⚪⚪
    🧪 PR contains tests
    🔒 No security concerns identified
    ⚡ Recommended focus areas for review

    Incomplete Documentation

    The Wheel class implementation has incomplete JSDoc comments. The scroll method parameters lack proper documentation compared to other methods in the file, which could make it difficult for developers to understand how to use this functionality correctly.

    class Wheel extends Device {
      /**
       * @param {string} id the device ID..
       */
      constructor(id) {
        super(Device.Type.WHEEL, id)
      }
    
      /**
       * Scrolls a page via the coordinates given
       * @param {number} x starting x coordinate
       * @param {number} y starting y coordinate
       * @param {number} deltaX Delta X to scroll to target
       * @param {number} deltaY Delta Y to scroll to target
       * @param {WebElement} origin element origin
       * @param {number} duration duration ratio be the ratio of time delta and duration
       * @returns {!Action} An action to scroll with this device.
       */
      scroll(x, y, deltaX, deltaY, origin, duration) {
        return {
          type: Action.Type.SCROLL,
          duration: duration,
          x: x,
          y: y,
          deltaX: deltaX,
          deltaY: deltaY,
          origin: origin,
        }
      }
    }
    Potential Bug

    The scroll method in the Actions class doesn't validate its parameters before passing them to the wheel_.scroll method, unlike other methods like dragAndDrop which perform input validation. This could lead to runtime errors if invalid parameters are provided.

    scroll(x, y, targetDeltaX, targetDeltaY, origin, duration) {
      return this.insert(this.wheel_, this.wheel_.scroll(x, y, targetDeltaX, targetDeltaY, origin, duration))
    }

    Copy link
    Contributor

    PR Code Suggestions ✨

    Explore these optional code suggestions:

    CategorySuggestion                                                                                                                                    Impact
    Possible issue
    Preserve all proxy exclusions

    The current implementation only takes the first element from the noProxy array.
    This will discard all other proxy exclusions. Consider preserving all noProxy
    entries by joining them into a comma-separated string instead of just taking the
    first element.

    javascript/selenium-webdriver/chromium.js [646-652]

     let proxy = caps.get(Capability.PROXY)
     if (proxy && Array.isArray(proxy.noProxy)) {
    -  proxy.noProxy = proxy.noProxy[0]
    +  proxy.noProxy = proxy.noProxy.join(',')
       if (!proxy.noProxy) {
         proxy.noProxy = undefined
       }
     }
    • Apply this suggestion
    Suggestion importance[1-10]: 9

    __

    Why: The current implementation discards all but the first element in the noProxy array, which would cause all other proxy exclusion rules to be lost. Joining the array elements into a comma-separated string preserves all exclusion rules, which is critical for proper proxy configuration. This is a significant bug fix that would prevent unexpected network behavior.

    High
    Handle rejected promise properly

    The test is not properly handling the rejected promise from the file detector.
    When handleFile returns a rejected promise, the test should verify that the
    error is properly caught and handled, but currently it just continues as if the
    promise was resolved successfully.

    javascript/selenium-webdriver/test/lib/webdriver_test.js [946-960]

     it('sendKeysWithAFileDetector_handlerError', function () {
       let executor = new FakeExecutor()
         .expect(CName.FIND_ELEMENT, {
           using: 'css selector',
           value: '*[id="foo"]',
         })
         .andReturnSuccess(WebElement.buildId('one'))
    -    .expect(CName.SEND_KEYS_TO_ELEMENT, {
    -      id: WebElement.buildId('one'),
    -      text: 'original/path',
    -      value: 'original/path'.split(''),
    -    })
    -    .andReturnSuccess()
         .end()
     
       let driver = executor.createDriver()
       let handleFile = function (d, path) {
         assert.strictEqual(driver, d)
         assert.strictEqual(path, 'original/path')
    -    return Promise.reject('unhandled file error')
    +    return Promise.reject(new Error('unhandled file error'))
       }
       driver.setFileDetector({ handleFile })
     
       return driver.findElement(By.id('foo')).sendKeys('original/', 'path')
    -})
    +    .then(fail, function(error) {
    +      assert.strictEqual(error.message, 'unhandled file error')
    +    })
    +}

    [To ensure code accuracy, apply this suggestion manually]

    Suggestion importance[1-10]: 8

    __

    Why: The suggestion correctly identifies a significant issue where the test doesn't properly verify error handling when the file detector returns a rejected promise. The improved code properly tests the error path, which is important for ensuring robust error handling in the WebDriver implementation.

    Medium
    Fix timing race condition

    This test has a race condition. It assumes that 75ms is less than half of 150ms
    and 175ms (75+100) is greater than 150ms, which may not always be true in all
    environments. The test should use more reliable timing mechanisms or mock the
    timer to ensure consistent behavior.

    javascript/selenium-webdriver/test/lib/webdriver_test.js [1136-1154]

     it('waits forever on a zero timeout', function () {
       let done = false
    -  setTimeout(() => (done = true), 150)
    +  let donePromise = new Promise(resolve => {
    +    setTimeout(() => {
    +      done = true
    +      resolve()
    +    }, 150)
    +  })
     
       let executor = new FakeExecutor()
       let driver = executor.createDriver()
       let waitResult = driver.wait(() => done, 0)
     
    -  return driver
    -    .sleep(75)
    -    .then(function () {
    -      assert.ok(!done)
    -      return driver.sleep(100)
    +  // Use a controlled timing approach
    +  return Promise.resolve()
    +    .then(() => {
    +      assert.ok(!done, 'Should not be done yet')
    +      return donePromise
         })
    -    .then(function () {
    -      assert.ok(done)
    +    .then(() => {
    +      assert.ok(done, 'Should be done now')
           return waitResult
         })
     })
    • Apply this suggestion
    Suggestion importance[1-10]: 7

    __

    Why: The suggestion addresses an important race condition in the test that relies on specific timing assumptions. The improved approach using a controlled promise-based timing mechanism makes the test more reliable and less susceptible to environment-specific timing variations.

    Medium
    General
    Fix parameter naming inconsistency

    The parameter names in the method signature don't match the names used in the
    function call. The parameters are named deltax and deltay in the documentation
    but targetDeltaX and targetDeltaY in the implementation, which could cause
    confusion for developers.

    javascript/selenium-webdriver/lib/input.js [844-852]

     /**
      * scrolls a page via the coordinates given
      * @param {number} x starting x coordinate
      * @param {number} y starting y coordinate
    - * @param {number} deltax delta x to scroll to target
    - * @param {number} deltay delta y to scroll to target
    + * @param {number} deltaX delta x to scroll to target
    + * @param {number} deltaY delta y to scroll to target
    + * @param {!Origin|!./webdriver.WebElement|undefined} origin the origin of the scroll
      * @param {number} duration duration ratio be the ratio of time delta and duration
      * @returns {!Actions} An action to scroll with this device.
      */
    -scroll(x, y, targetDeltaX, targetDeltaY, origin, duration) {
    -  return this.insert(this.wheel_, this.wheel_.scroll(x, y, targetDeltaX, targetDeltaY, origin, duration))
    +scroll(x, y, deltaX, deltaY, origin, duration) {
    +  return this.insert(this.wheel_, this.wheel_.scroll(x, y, deltaX, deltaY, origin, duration))
     }

    [To ensure code accuracy, apply this suggestion manually]

    Suggestion importance[1-10]: 7

    __

    Why: The suggestion correctly identifies a parameter naming inconsistency between the JSDoc comments and the actual implementation. This is important for code maintainability and developer understanding, as the mismatch between documentation and implementation could lead to confusion.

    Medium
    Prevent promise memory leak

    The test creates a deferred promise but never resolves or rejects it, which
    could lead to memory leaks. Even though the test is checking timeout behavior,
    it's good practice to clean up by resolving or rejecting the promise after the
    test completes.

    javascript/selenium-webdriver/test/lib/webdriver_test.js [1206-1218]

     it('promise that does not resolve before timeout', function () {
       let d = defer()
     
       let executor = new FakeExecutor()
       let driver = executor.createDriver()
    -  return driver.wait(d.promise, 5).then(fail, (e) => {
    -    assert.ok(e instanceof error.TimeoutError, 'Unexpected error: ' + e)
    -    assert.ok(
    -      /Timed out waiting for promise to resolve after \d+ms/.test(e.message),
    -      'unexpected error message: ' + e.message,
    -    )
    -  })
    +  return driver.wait(d.promise, 5)
    +    .then(fail, (e) => {
    +      assert.ok(e instanceof error.TimeoutError, 'Unexpected error: ' + e)
    +      assert.ok(
    +        /Timed out waiting for promise to resolve after \d+ms/.test(e.message),
    +        'unexpected error message: ' + e.message,
    +      )
    +    })
    +    .finally(() => {
    +      // Clean up the promise to avoid potential memory leaks
    +      d.resolve();
    +    })
     })
    • Apply this suggestion
    Suggestion importance[1-10]: 5

    __

    Why: The suggestion correctly identifies a potential memory leak where a deferred promise is created but never resolved. While this is a valid improvement for test hygiene, it's a relatively minor issue compared to the other suggestions since the test process would end anyway.

    Low
    Learned
    best practice
    Initialize all object properties in a single step or within the same scope to improve code clarity and reduce null reference risks

    The defer() function creates an object and then assigns properties to it in
    multiple steps. This approach can be improved by using a more concise pattern
    that initializes all properties at once, reducing the risk of null reference
    issues and making the code more maintainable.

    javascript/selenium-webdriver/test/lib/webdriver_test.js [40-47]

     function defer() {
    -  let d = {}
    -  let promise = new Promise((resolve, reject) => {
    -    Object.assign(d, { resolve, reject })
    +  let d = { promise: null }
    +  d.promise = new Promise((resolve, reject) => {
    +    d.resolve = resolve
    +    d.reject = reject
       })
    -  d.promise = promise
       return d
     }
    • Apply this suggestion
    Suggestion importance[1-10]: 6
    Low
    • More

    @harsha509
    Copy link
    Member Author

    LGTM!

    @harsha509 harsha509 merged commit c6210c1 into SeleniumHQ:trunk Mar 25, 2025
    35 of 36 checks passed
    Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
    Projects
    None yet
    Development

    Successfully merging this pull request may close these issues.

    1 participant