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

Add ability to customize automatic update channels for add-ons #17597

Draft
wants to merge 8 commits into
base: master
Choose a base branch
from

Conversation

seanbudd
Copy link
Member

@seanbudd seanbudd commented Jan 8, 2025

Link to issue number:

Part of #3208
Closes #15261

Summary of the issue:

Add-ons currently only automatically update to the same channel with automatic updates enabled
This is quite restrictive as many people may want the following scenarios:

  • disabling specific add-ons from update, with automatic updates enabled by default
  • enabling only specific add-ons from update, disabling automatic updates by default
  • sticking to the latest version regardless of channel
  • migrating back to a stable version from a beta/dev version
  • sticking to the latest beta or dev version only
  • migrating from an externally installed version

Description of user facing changes

Added an "update channel" submenu for add-ons on the installed/update tabs of the add-on store.
This allows selecting a specific set of update channels on a per add-on basis.

Added an item to the add-on store settings panel for selecting the default update channel for all add-ons

Description of development approach

  • Created an add-on store settings cache, which also stores add-on specific information.
  • Migrated the "showWarning" setting to add-on store specific settings, so it can be saved independently of config profiles

Testing strategy:

Known issues with pull request:

Code Review Checklist:

  • Documentation:
    • Change log entry
    • User Documentation
    • Developer / Technical Documentation
    • Context sensitive help for GUI changes
  • Testing:
    • Unit tests
    • System (end to end) tests
    • Manual testing
  • UX of all users considered:
    • Speech
    • Braille
    • Low Vision
    • Different web browsers
    • Localization in other languages / culture than English
  • API is compatible with existing add-ons.
  • Security precautions taken.

Summary by CodeRabbit

Release Notes

  • New Features

    • Added advanced add-on update channel management
    • Users can now set default and per-addon update channels
    • Introduced more granular control over automatic add-on updates
  • Improvements

    • Enhanced add-on store settings and configuration options
    • Refined update notification logic for add-ons
    • Improved flexibility in managing addon update preferences
  • User Experience

    • New options to customize add-on update behavior
    • More intuitive add-on channel selection
    • Simplified update channel management interface

@seanbudd seanbudd requested review from a team as code owners January 8, 2025 00:32
@seanbudd
Copy link
Member Author

seanbudd commented Jan 8, 2025

@coderabbitai review

Copy link
Contributor

coderabbitai bot commented Jan 8, 2025

Walkthrough

This pull request introduces a comprehensive enhancement to the add-on store settings and update management in NVDA. The changes include a new settings management system for add-ons, introducing more flexible update channel controls, and modifying how warning dialogs and update preferences are handled. The implementation spans multiple files, introducing new classes for settings management, update channel definitions, and updating various GUI components to support these new features.

Changes

File Change Summary
source/addonStore/dataManager.py Added storeSettings to manage add-on store configurations, updated terminate method to save settings
source/addonStore/models/channel.py Introduced UpdateChannel enum with various update channel constants and methods
source/addonStore/settings.py Added _AddonSettings and _AddonStoreSettings classes for managing add-on store settings
source/config/configSpec.py Removed showWarning, added defaultUpdateChannel configuration option
source/config/profileUpgradeSteps.py Added upgradeConfigFrom_14_to_15 to handle migration of warning settings
source/gui/addonStoreGui/controls/actions.py Added _UpdateChannelSubMenu for managing update channel actions
source/gui/settingsDialogs.py Added combo box for selecting default update channel

Assessment against linked issues

Objective Addressed Explanation
Persistent "Don't show again" setting [#15261]
Independent of profile settings [#15261]
Always save warning preference independently [#15261]

Possibly related PRs

Suggested labels

conceptApproved, merge-early

Suggested reviewers

  • Qchristensen

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 generate docstrings to generate docstrings for this PR. (Beta)
  • @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.

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
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: 2

🧹 Nitpick comments (7)
source/addonStore/settings.py (1)

37-56: Consider adding file locking mechanism.

The settings file could potentially be accessed concurrently by multiple NVDA instances. Consider implementing file locking to prevent race conditions.

 def __init__(self):
+    import portalocker
     self._storeSettingsFile = os.path.join(
         NVDAState.WritePaths.addonStoreDir,
         self._storeSettingsFilename,
     )
+    self._file_lock = portalocker.Lock(self._storeSettingsFile, timeout=1)
     self._showWarning = True
     self._addonSettings = {}
     self.load()
source/gui/addonStoreGui/controls/actions.py (1)

100-128: Consider handling radio item group scope.

The radio items are added to the menu without explicitly managing their group scope. This could lead to unexpected behavior if multiple update channel submenus are shown simultaneously.

 def _insertToContextMenu(self, action: AddonUpdateChannelActionVM, prevActionIndex: int):
+    # Create a new radio group for each submenu
+    if prevActionIndex == 0:
+        self._contextMenu.Break()  # Start a new radio group
     self._actionMenuItemMap[action] = self._contextMenu.InsertRadioItem(
         prevActionIndex,
         id=-1,
         item=action.displayName,
     )
source/addonStore/dataManager.py (1)

356-359: Consider documenting the configuration setting

The TODO comment indicates a future enhancement for incompatible updates. Consider documenting the configuration setting that will control this behavior.

tests/unit/test_config.py (1)

1071-1081: Incorrect docstring for test_profileWithShowWarningSetTrue.

The docstring states "Test that a profile with showWarning not set is correctly upgraded" but the test actually verifies migration of showWarning=True.

-        """Test that a profile with showWarning not set is correctly upgraded."""
+        """Test that a profile with showWarning set to True is correctly upgraded."""
source/addonStore/models/channel.py (1)

122-145: Consider adding error handling for invalid addon channel.

The method handles invalid update channels but doesn't validate the addon channel parameter.

 def _availableChannelsForAddonWithChannel(self, addonChannel: Channel) -> set[Channel]:
     """Return the available update channels for an addon with the given channel and the update channel set."""
+    if not isinstance(addonChannel, Channel):
+        raise ValueError(f"Invalid addon channel: {addonChannel}")
     if self == UpdateChannel.DEFAULT:
         channel = UpdateChannel(config.conf["addonStore"]["defaultUpdateChannel"])
user_docs/en/changes.md (2)

9-12: Consider adding version numbers to highlight text

The highlight text could be enhanced by including version numbers, e.g.:

- Add ability to customize automatic update channels for add-ons
+ Add ability to customize automatic update channels for add-ons (v17597)

169-170: Inconsistent whitespace before section heading

There is an extra blank line before the "Changes for Developers" section that is inconsistent with other section spacing.

-
### Changes for Developers
+### Changes for Developers
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 1edb075 and aa10ee4.

📒 Files selected for processing (13)
  • source/addonStore/dataManager.py (5 hunks)
  • source/addonStore/models/channel.py (3 hunks)
  • source/addonStore/settings.py (1 hunks)
  • source/config/configSpec.py (3 hunks)
  • source/config/profileUpgradeSteps.py (2 hunks)
  • source/gui/addonStoreGui/controls/actions.py (7 hunks)
  • source/gui/addonStoreGui/controls/messageDialogs.py (1 hunks)
  • source/gui/addonStoreGui/controls/storeDialog.py (1 hunks)
  • source/gui/addonStoreGui/viewModels/action.py (2 hunks)
  • source/gui/settingsDialogs.py (2 hunks)
  • tests/unit/test_config.py (3 hunks)
  • user_docs/en/changes.md (2 hunks)
  • user_docs/en/userGuide.md (3 hunks)
🧰 Additional context used
📓 Path-based instructions (13)
source/gui/addonStoreGui/controls/messageDialogs.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/config/configSpec.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/gui/addonStoreGui/controls/storeDialog.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/addonStore/models/channel.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/config/profileUpgradeSteps.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/gui/addonStoreGui/viewModels/action.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/addonStore/settings.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/gui/addonStoreGui/controls/actions.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/gui/settingsDialogs.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

source/addonStore/dataManager.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

tests/unit/test_config.py (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.py: _, pgettext, ngettext, and ngettext are defined globally, errors for this being undefined can be ignored.

user_docs/en/userGuide.md (2)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.md: Focus on grammar, spelling, and punctuation. Also consider readability and clarity of contents. Ensure that changes follow the style of one sentence per line, suggest changes if this is not met.

user_docs/en/changes.md (3)

Pattern **/*: Focus on code smells, logic errors, edge cases, missing test cases, security flaws and serious issues. Avoid commenting on minor issues such as linting, formatting and style issues. This project uses tabs instead of spaces, do not suggest usage of spaces over tabs. Are there any 'red flags' in this code that might warrant closer investigation from a security standpoint? Explain what makes them suspicious. When providing code suggestions, particularly when requested, ensure GitHub's suggestion format is used, i.e.: suggestion <code changes>


Pattern **/*.md: Focus on grammar, spelling, and punctuation. Also consider readability and clarity of contents. Ensure that changes follow the style of one sentence per line, suggest changes if this is not met.


Pattern user_docs/en/changes.md: Ensure each change log entry references an issue or pull request number. Change log entries can also include a reference to a GitHub author. Examples of valid change log entries: * Item with sub-items (#123, @username): * sub-item * bar (#342) * Item with. Multiple lines. (#143)

⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: Analyze (python)
🔇 Additional comments (26)
source/addonStore/settings.py (3)

16-24: LGTM! Well-structured dataclass implementation.

The dataclass is well-documented and follows best practices with appropriate default values.


26-34: Consider tracking the TODO items.

The TODO comments indicate plans to migrate state tracking from addonHandler. This could be important for maintaining a consistent state management approach.

Would you like me to help create GitHub issues to track these TODO items for future implementation?


105-131: LGTM! Well-implemented settings management methods.

The settings management methods are well-structured with appropriate error handling and documentation.

source/gui/addonStoreGui/viewModels/action.py (1)

174-188: LGTM! Well-implemented action view model.

The implementation follows the existing patterns and is well-documented. The use of lambda aVM: True for validCheck is appropriate since the channel can always be changed from the installed/update tabs.

source/gui/addonStoreGui/controls/actions.py (2)

129-133: LGTM! Well-implemented action list.

The _actions property correctly creates update channel actions for all available channels.


152-164: LGTM! Appropriate menu integration.

The update channel submenu is correctly integrated into the context menu with proper filtering based on status.

source/config/configSpec.py (1)

340-343: LGTM! Well-documented configuration setting.

The defaultUpdateChannel setting is well-documented with clear value ranges and appropriate validation. The comment explaining the 1-based indexing is particularly helpful.

source/addonStore/dataManager.py (4)

47-47: LGTM!

The import statement is correctly placed and follows the project's import organization pattern.


103-103: LGTM!

The storeSettings initialization is placed appropriately in the constructor and follows the initialization pattern of other instance variables.


116-116: LGTM!

The save() call is correctly placed in the terminate method to ensure settings persistence.


364-376: Verify the update channel selection logic

The new update channel selection logic looks correct but complex. Let's verify the implementation:

  1. Gets installed channel from store data or defaults to EXTERNAL
  2. Retrieves addon-specific update channel settings
  3. Determines available update channels
  4. Checks if addon's channel is in available channels

Run this script to verify the update channel selection logic:

✅ Verification successful

Update channel selection logic is correctly implemented

The implementation in UpdateChannel._availableChannelsForAddonWithChannel properly handles all update scenarios:

  • DEFAULT: Uses user's preferred channel from config
  • SAME: Allows updates only within current channel
  • ANY: Allows updates across all channels
  • NO_UPDATE: Prevents any updates
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the update channel selection implementation
# Test: Search for the method that determines available channels
ast-grep --pattern $'_availableChannelsForAddonWithChannel($_, $_) {
  $$$
}'

Length of output: 78


Script:

#!/bin/bash
# Find Channel related files
fd -t f "channel" -i

# Search for update channel related code and the method name
rg -i "availablechannelsforaddonwithchannel|class.*channel" -A 5

# Backup search for the method name without class context
rg "_availableChannelsForAddonWithChannel" -A 5

Length of output: 4442


Script:

#!/bin/bash
# Get the complete UpdateChannel class implementation
rg -U "class UpdateChannel.*?(?=class|\z)" -P source/addonStore/models/channel.py

# Get any related helper methods or constants
rg "def.*Channel" source/addonStore/models/channel.py -A 10

Length of output: 2416

source/gui/addonStoreGui/controls/storeDialog.py (1)

55-55: LGTM!

The change correctly uses the new settings management system to access the warning display preference.

source/config/profileUpgradeSteps.py (1)

476-485: LGTM!

The upgrade function correctly:

  1. Checks for existence of old setting
  2. Migrates the value to the new settings system
  3. Removes the old setting
  4. Includes appropriate debug logging
source/gui/addonStoreGui/controls/messageDialogs.py (1)

358-358: LGTM!

The change correctly updates the warning preference using the new settings management system.

tests/unit/test_config.py (3)

1049-1050: LGTM: Test class name follows naming convention.

The test class name clearly indicates it's testing the upgrade from config version 14 to 15.


1050-1058: LGTM: Test method verifies default profile behavior.

The test correctly verifies that the default profile does not set the showWarning attribute as a boolean in the store settings.


1059-1070: LGTM: Test method verifies migration of showWarning=False.

The test properly verifies that when showWarning is explicitly set to False in the profile, it is correctly migrated to the store settings.

source/gui/settingsDialogs.py (3)

3265-3266: LGTM: Clean import of UpdateChannel.

The import is placed with other imports and follows PEP 8 guidelines.


3267-3280: LGTM: Well-structured combo box setup.

The code:

  • Uses clear label text
  • Correctly filters out UpdateChannel.DEFAULT
  • Properly handles accessibility with help event binding

3358-3361: LGTM: Proper handling of 1-based indexing.

The code correctly:

  • Adds 1 to convert from 0-based UI index to 1-based config value
  • Includes clear comment explaining the adjustment
source/addonStore/models/channel.py (2)

58-86: LGTM: Well-documented UpdateChannel enum.

The enum:

  • Has clear docstrings for each value
  • Uses meaningful names
  • Follows consistent ordering

87-98: LGTM: Smart handling of default channel display string.

The code:

  • Avoids recursive dependency
  • Includes assertion for safety
  • Properly handles localization
user_docs/en/userGuide.md (2)

3081-3082: LGTM! Clear explanation of update notification behavior.

The text clearly explains that add-on update notifications will only occur for add-ons with updates available within the same channel by default. This helps users understand the expected behavior.


3094-3115: LGTM! Well-structured documentation of update channel options.

The section provides a comprehensive explanation of the default update channel setting, with:

  • Clear heading and introduction
  • Well-formatted table of available options
  • Detailed descriptions of each option's behavior
user_docs/en/changes.md (2)

Line range hint 1-169: LGTM! Well-structured changelog format.

The changelog follows a clear and consistent format with major version releases organized chronologically, each containing standardized sections for New Features, Changes, Bug Fixes and Changes for Developers. The Markdown formatting is clean and readable.


Line range hint 1-10000: Verify all GitHub issue references are still valid

The changelog contains many GitHub issue references like (#xxxx). It would be good to verify these links still work and point to the correct issues.

✅ Verification successful

Issue references in changelog are valid and accessible

Based on testing a sample of issue references from the changelog:

  • Found 1990 unique issue references
  • Tested 10 sample issues (5 oldest and 5 newest)
  • All tested issues were valid and accessible on GitHub
  • References include both issues and pull requests
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if GitHub issue references are valid
# Extract issue numbers and check if they exist via GitHub API
rg -o '#\(\d+\)' | while read -r match; do
  issue=$(echo "$match" | tr -dc '0-9')
  gh api "/repos/nvaccess/nvda/issues/$issue" >/dev/null 2>&1 || echo "Invalid issue: $match"
done

Length of output: 424


Script:

#!/bin/bash
# Extract issue numbers from changes.md and verify them
echo "Checking GitHub issue references..."

# Extract unique issue numbers from changes.md
echo "Extracting issue numbers..."
rg -o '#\((\d+)\)' changes.md --replace '$1' | sort -u > issues.txt

# Check first few issues as a sample (limit API calls)
head -n 5 issues.txt | while read -r issue; do
  echo "Checking issue #$issue..."
  if ! gh api "/repos/nvaccess/nvda/issues/$issue" --jq '.html_url' 2>/dev/null; then
    echo "Warning: Issue #$issue may be invalid"
  fi
done

echo "Sample check complete. Total unique issues found: $(wc -l < issues.txt)"

Length of output: 414


Script:

#!/bin/bash
# Extract issue numbers from changes.md and verify them
CHANGES_FILE="user_docs/en/changes.md"

echo "Checking GitHub issue references..."

if [ ! -f "$CHANGES_FILE" ]; then
    echo "Error: Cannot find $CHANGES_FILE"
    exit 1
fi

# Extract unique issue numbers from changes.md
echo "Extracting issue numbers..."
rg -o '#\((\d+)\)' "$CHANGES_FILE" --replace '$1' | sort -u > issues.txt

# Show total count
echo "Found $(wc -l < issues.txt) unique issue references"

# Check first 5 issues as a sample
echo "Checking sample of issues..."
head -n 5 issues.txt | while read -r issue; do
    echo -n "Checking #$issue: "
    if gh api "/repos/nvaccess/nvda/issues/$issue" --jq '.html_url' 2>/dev/null; then
        echo "Valid"
    else
        echo "Warning: May be invalid"
    fi
done

Length of output: 483


Script:

#!/bin/bash
# First locate and verify the changes file
echo "Searching for changes.md file..."
fd changes.md

echo -e "\nTrying alternate locations..."
ls -l user_docs/en/ 2>/dev/null || echo "user_docs/en/ not found"
ls -l docs/ 2>/dev/null || echo "docs/ not found"

echo -e "\nTesting with a sample file..."
# Try to extract from the first changes file found
CHANGES_FILE=$(fd changes.md | head -n 1)
if [ -n "$CHANGES_FILE" ]; then
    echo "Found file: $CHANGES_FILE"
    echo "First few lines:"
    head -n 5 "$CHANGES_FILE"
    echo -e "\nTesting issue extraction:"
    # Try simpler pattern
    grep -o "#([0-9]\+)" "$CHANGES_FILE" | head -n 5 || echo "No matches with pattern #([0-9]+)"
fi

Length of output: 1443


Script:

#!/bin/bash
CHANGES_FILE="user_docs/en/changes.md"
echo "Analyzing $CHANGES_FILE..."

# Show a sample of the content to verify issue reference format
echo "Sample content (first occurrence of an issue reference):"
grep -A 1 -m 1 "#(" "$CHANGES_FILE"

echo -e "\nTrying to extract issue numbers..."
# Try multiple patterns to ensure we catch the references
echo "Pattern 1 results:"
grep -o "#([0-9]*)" "$CHANGES_FILE" | head -n 5

echo -e "\nPattern 2 results:"
grep -o "#([[:digit:]]*)" "$CHANGES_FILE" | head -n 5

echo -e "\nPattern 3 results:"
perl -ne 'print "$&\n" while /#\(\d+\)/g' "$CHANGES_FILE" | head -n 5

echo -e "\nTotal references found:"
perl -ne 'print "$&\n" while /#\(\d+\)/g' "$CHANGES_FILE" | wc -l

Length of output: 871


Script:

#!/bin/bash
CHANGES_FILE="user_docs/en/changes.md"

echo "File info:"
ls -l "$CHANGES_FILE"

echo -e "\nFirst 10 lines of content:"
head -n 10 "$CHANGES_FILE"

echo -e "\nSearching for lines containing '#':"
grep -n "#" "$CHANGES_FILE" | head -n 5

echo -e "\nSearching for lines containing numbers in parentheses:"
grep -n "([0-9]" "$CHANGES_FILE" | head -n 5

echo -e "\nSearching for specific issue reference format:"
grep -n "(\#[0-9]" "$CHANGES_FILE" | head -n 5

Length of output: 2147


Script:

#!/bin/bash
CHANGES_FILE="user_docs/en/changes.md"

echo "Extracting all unique issue numbers..."
# Extract numbers between (# and )
perl -ne 'print "$1\n" while /\(#(\d+)\)/g' "$CHANGES_FILE" | sort -un > issue_numbers.txt

echo "Total unique issues found: $(wc -l < issue_numbers.txt)"

echo -e "\nChecking random sample of 5 issues..."
# Check 5 random issues
shuf -n 5 issue_numbers.txt | while read -r issue; do
    echo -n "Issue #$issue: "
    if gh api "/repos/nvaccess/nvda/issues/$issue" --jq '.html_url' 2>/dev/null; then
        echo "Valid"
    else
        echo "Not found or inaccessible"
    fi
done

Length of output: 521


Script:

#!/bin/bash
CHANGES_FILE="user_docs/en/changes.md"

echo "Extracting all unique issue numbers..."
perl -ne 'print "$1\n" while /\(#(\d+)\)/g' "$CHANGES_FILE" | sort -un > issue_numbers.txt
TOTAL=$(wc -l < issue_numbers.txt)
echo "Total unique issues found: $TOTAL"

echo -e "\nChecking sample of first 5 issues..."
head -n 5 issue_numbers.txt | while read -r issue; do
    echo -n "Issue #$issue: "
    if gh api "/repos/nvaccess/nvda/issues/$issue" --jq '.html_url' 2>/dev/null; then
        echo "Valid"
    else
        echo "Not found or inaccessible"
    fi
done

echo -e "\nChecking sample of last 5 issues..."
tail -n 5 issue_numbers.txt | while read -r issue; do
    echo -n "Issue #$issue: "
    if gh api "/repos/nvaccess/nvda/issues/$issue" --jq '.html_url' 2>/dev/null; then
        echo "Valid"
    else
        echo "Not found or inaccessible"
    fi
done

Length of output: 2299

source/addonStore/settings.py Outdated Show resolved Hide resolved
source/addonStore/settings.py Outdated Show resolved Hide resolved
@@ -13,7 +13,7 @@
#: provide an upgrade step (@see profileUpgradeSteps.py). An upgrade step does not need to be added when
#: just adding a new element to (or removing from) the schema, only when old versions of the config
#: (conforming to old schema versions) will not work correctly with the new schema.
latestSchemaVersion = 14
latestSchemaVersion = 15
Copy link
Member

Choose a reason for hiding this comment

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

This may need to be increased if #17505 is merged first.

# UpdateChannel values:
# same channel (default), any channel, do not update, stable, beta & dev, beta, dev
# Not 0 based as other usages of UpdateChannel's 0-value is used to refer to this default fallback value.
defaultUpdateChannel = integer(1, 7, default=1)
Copy link
Member

Choose a reason for hiding this comment

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

I am concerned about this being here, but add-on specific update channels being global. It feels unintuitive to me, and like it may lead to unexpected results (Ex. add-ons updating to a different channel depending on config profile in use).

Copy link
Member Author

Choose a reason for hiding this comment

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

profiles shouldn't matter as addonStore settings are global across profiles

if "addonStore" in profile and "showWarning" in profile["addonStore"]:
from addonStore.dataManager import addonDataManager

addonDataManager.storeSettings.showWarning = profile["addonStore"].as_bool("showWarning")
Copy link
Member

Choose a reason for hiding this comment

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

What happens if the config contains an invalid value?

Copy link
Member Author

Choose a reason for hiding this comment

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

I believe when there is an invalid config it fails to load in NVDA and resets the users config.

Copy link
Member

Choose a reason for hiding this comment

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

Validation happens after upgrade (the old config could be invalid according to the new schema, so it logically has to happen in this order), so there's no guarantee that you'll get a bool.

Comment on lines +26 to +34
# TODO: migrate enabled/disabled/blocked state tracking
# from addonHandler.AddonState/AddonStateCategory to here.
# The set based state tracking could be replaced by maintaining state data on each add-on.
#
# blocked: bool = False
# """Whether the add-on is blocked from being running due to incompatibility."""
#
# disabled: bool = False
# """Whether the add-on is disabled."""
Copy link
Member

Choose a reason for hiding this comment

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

Has this intentionally not been done?

Copy link
Member Author

Choose a reason for hiding this comment

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

yep, out of scope for this PR

Copy link
Member

Choose a reason for hiding this comment

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

Is there an issue for it? If not, can you create one?

source/addonStore/settings.py Outdated Show resolved Hide resolved
tests/unit/test_config.py Outdated Show resolved Hide resolved
user_docs/en/userGuide.md Outdated Show resolved Hide resolved
Comment on lines 3106 to 3114
| Option | Behaviour |
|---|---|
| Same | Add-ons will remain on their channel |
| Any | Add-ons will always automatically update to the latest version, regardless of channel |
| Do not update | Add-ons will not automatically update by default, you must enable them individually |
| Stable | Add-ons will only automatically update to stable versions |
| Beta/Dev | Add-ons will only automatically update to beta or dev versions |
| Beta | Add-ons will only automatically update to beta versions |
| Dev | Add-ons will only automatically update to dev versions |
Copy link
Member

Choose a reason for hiding this comment

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

I don't think the wording here is very clear:

  1. Get rid of the words "only" and "automatically". The options can be overridden on a per add-on basis, so there is no "only", and nothing will happen "automatically" if the user has automatic updates disabled.

Copy link
Member Author

Choose a reason for hiding this comment

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

I think removing "automatically" will make it unclear that this applies to automatic updates, i.e., user's may assume that changing these settings will affect their ability to manually update add-ons.

user_docs/en/userGuide.md Outdated Show resolved Hide resolved
user_docs/en/userGuide.md Show resolved Hide resolved
source/addonStore/models/channel.py Outdated Show resolved Hide resolved
user_docs/en/userGuide.md Show resolved Hide resolved
# UpdateChannel values:
# same channel (default), any channel, do not update, stable, beta & dev, beta, dev
# Not 0 based as other usages of UpdateChannel's 0-value is used to refer to this default fallback value.
defaultUpdateChannel = integer(1, 7, default=1)
Copy link
Member Author

Choose a reason for hiding this comment

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

profiles shouldn't matter as addonStore settings are global across profiles

if "addonStore" in profile and "showWarning" in profile["addonStore"]:
from addonStore.dataManager import addonDataManager

addonDataManager.storeSettings.showWarning = profile["addonStore"].as_bool("showWarning")
Copy link
Member Author

Choose a reason for hiding this comment

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

I believe when there is an invalid config it fails to load in NVDA and resets the users config.

source/gui/settingsDialogs.py Outdated Show resolved Hide resolved
source/addonStore/settings.py Show resolved Hide resolved
user_docs/en/userGuide.md Outdated Show resolved Hide resolved
user_docs/en/userGuide.md Outdated Show resolved Hide resolved
user_docs/en/userGuide.md Outdated Show resolved Hide resolved
tests/unit/test_config.py Show resolved Hide resolved
@seanbudd seanbudd requested a review from SaschaCowley January 16, 2025 03:21
This option sets the default update channel for all add-ons.
You can also change the update channel for a [specific add-on individually from the Add-on Store](#AddonStoreUpdateChannel).

| . {.hideHeaderRow} |.|
|---|---|
| Options | Same (Default), Any, Do not update, Stable, Beta/Dev, Beta, Dev |
| Options | Same (Default), Any, Do not update, Stable, Beta or dev, Beta, Dev |
| Default | Same |

| Option | Behaviour |
|---|---|
| Same | Add-ons will remain on their channel |
| Any | Add-ons will always automatically update to the latest version, regardless of channel |
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
| Any | Add-ons will always automatically update to the latest version, regardless of channel |
| Any | Add-ons will automatically update to the latest version, regardless of channel |

@@ -3767,10 +3767,10 @@ From an add-on's actions menu, using the submenu "Update channel", you can modif
| Same | Add-on will remain on the same channel |
| Any | Add-on will always automatically update to the latest version, regardless of channel |
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
| Any | Add-on will always automatically update to the latest version, regardless of channel |
| Any | Add-on will automatically update to the latest version, regardless of channel |

Comment on lines +26 to +34
# TODO: migrate enabled/disabled/blocked state tracking
# from addonHandler.AddonState/AddonStateCategory to here.
# The set based state tracking could be replaced by maintaining state data on each add-on.
#
# blocked: bool = False
# """Whether the add-on is blocked from being running due to incompatibility."""
#
# disabled: bool = False
# """Whether the add-on is disabled."""
Copy link
Member

Choose a reason for hiding this comment

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

Is there an issue for it? If not, can you create one?

Comment on lines +58 to +61
settingsDict = {
"showWarning": True,
"addonSettings": {},
}
Copy link
Member

Choose a reason for hiding this comment

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

In _AddonStoreSettings.__init__, you already define the default value of _showWarning and _addonSettings.

self._showWarning = True
self._addonSettings = {}

So it probably makes sense to

  1. Return if a settings cache is not found (
    except FileNotFoundError:
    pass
    )
  2. Return if the settings cache is not JSON (
    except Exception:
    log.exception("Invalid add-on store settings")
    if NVDAState.shouldWriteToDisk():
    os.remove(self._storeSettingsFile)
    )
  3. Return if the types are invalid (
    except (KeyError, ValueError):
    log.exception(f"Invalid add-on store cache:\n{settingsDict}")
    if NVDAState.shouldWriteToDisk():
    os.remove(self._storeSettingsFile)
    )

with open(self._storeSettingsFile, "r", encoding="utf-8") as storeSettingsFile:
settingsDict = json.load(storeSettingsFile)
except FileNotFoundError:
pass
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
pass
return

except (KeyError, ValueError):
log.exception(f"Invalid add-on store cache:\n{settingsDict}")
if NVDAState.shouldWriteToDisk():
os.remove(self._storeSettingsFile)
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
os.remove(self._storeSettingsFile)
os.remove(self._storeSettingsFile)
return

source/addonStore/settings.py Show resolved Hide resolved
source/addonStore/settings.py Show resolved Hide resolved
source/addonStore/settings.py Show resolved Hide resolved
if "addonStore" in profile and "showWarning" in profile["addonStore"]:
from addonStore.dataManager import addonDataManager

addonDataManager.storeSettings.showWarning = profile["addonStore"].as_bool("showWarning")
Copy link
Member

Choose a reason for hiding this comment

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

Validation happens after upgrade (the old config could be invalid according to the new schema, so it logically has to happen in this order), so there's no guarantee that you'll get a bool.

@SaschaCowley SaschaCowley marked this pull request as draft January 16, 2025 05:29
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.

Warning dialog of Add-on Store is shown again although "Don't show again" checkbox has been checked
2 participants