fix(zsh): escape parentheses and brackets in completion descriptions#559
fix(zsh): escape parentheses and brackets in completion descriptions#559
Conversation
zsh's `_describe` function interprets parentheses as glob qualifiers and brackets as character classes, causing errors like "unknown file attribute" and "unknown sort specifier" when completion descriptions contain these characters. Fixes #558 Co-Authored-By: Claude Opus 4.6 (1M context) <[email protected]>
Summary of ChangesHello, I'm Gemini Code Assist1! I'm currently reviewing this pull request and will post my feedback shortly. In the meantime, here's a summary to help you and other reviewers quickly get up to speed! This pull request addresses an issue where Zsh's Highlights
Using Gemini Code AssistThe full guide for Gemini Code Assist can be found on our documentation page, here are some quick tips. Invoking Gemini You can request assistance from Gemini at any point by creating a comment using either
Customization To customize Gemini Code Assist for GitHub experience, repository maintainers can create a configuration file and/or provide a custom code review style guide (such as PEP-8 for Python) by creating and adding files to a Limitations & Feedback Gemini Code Assist may make mistakes. Please leave feedback on any instances where its feedback is incorrect or counter productive. You can react with 👍 and 👎 on @gemini-code-assist comments. If you're interested in giving your feedback about your experience with Gemini Code Assist for GitHub and other Google products, sign up here. Footnotes
|
Codecov Report❌ Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #559 +/- ##
==========================================
+ Coverage 70.04% 70.07% +0.03%
==========================================
Files 48 48
Lines 7080 7088 +8
Branches 7080 7088 +8
==========================================
+ Hits 4959 4967 +8
- Misses 1322 1323 +1
+ Partials 799 798 -1 ☔ View full report in Codecov by Sentry. 🚀 New features to boost your workflow:
|
Greptile SummaryThis PR fixes a zsh completion bug where unescaped
Confidence Score: 5/5
Important Files Changed
Sequence DiagramsequenceDiagram
participant Shell as zsh completion script
participant CW as complete-word (--shell zsh)
participant Script as run script (sh -c)
Shell->>CW: complete-word --shell zsh -f spec -- words
CW->>Script: sh -c "echo connect\\:server:Connect server (Hot Reload)\n..."
Script-->>CW: connect\:server:Connect server (Hot Reload)\ntest\:unit:Run tests [fast]\nbuild:Build project
Note over CW: parse: split on first unescaped colon,<br/>unescape \\: → : in value & description
Note over CW: zsh_escape(value) → escape :, (, ), [, ]<br/>zsh_escape(description) → same
CW-->>Shell: connect\:server:Connect server \(Hot Reload\)\ntest\:unit:Run tests \[fast\]\nbuild:Build project
Shell->>Shell: completions+=("$line") for each line
Shell->>Shell: _describe 'completions' completions -S ''
Reviews (1): Last reviewed commit: "fix(zsh): escape parentheses and bracket..." | Re-trigger Greptile |
There was a problem hiding this comment.
Code Review
This pull request introduces a new zsh_escape function to correctly handle special characters like colons, parentheses, and brackets in zsh completion output, addressing an issue where these characters were misinterpreted as glob qualifiers or character classes. A new test case and example file have been added to validate this functionality. A performance improvement opportunity was identified for the zsh_escape function, suggesting a single-pass iteration instead of chained replace calls to reduce string allocations.
| s.replace(':', "\\:") | ||
| .replace('(', "\\(") | ||
| .replace(')', "\\)") | ||
| .replace('[', "\\[") | ||
| .replace(']', "\\]") |
There was a problem hiding this comment.
While this implementation is correct, chaining replace calls can be inefficient as it may create intermediate String allocations and perform multiple passes over the string. A more performant approach would be to iterate over the string once and build the escaped string in a single pass. This avoids the overhead of multiple allocations and iterations, especially if the input strings can be long.
| s.replace(':', "\\:") | |
| .replace('(', "\\(") | |
| .replace(')', "\\)") | |
| .replace('[', "\\[") | |
| .replace(']', "\\]") | |
| let mut escaped = String::with_capacity(s.len()); | |
| for c in s.chars() { | |
| match c { | |
| ':' | '(' | ')' | '[' | ']' => { | |
| escaped.push('\\'); | |
| escaped.push(c); | |
| } | |
| _ => escaped.push(c), | |
| } | |
| } | |
| escaped |
### 🚀 Features - Support env-backed choices with `choices env=...` by [@mustafa0x](https://github.com/mustafa0x) in [#548](#548) ### 🐛 Bug Fixes - **(zsh)** escape parentheses and brackets in completion descriptions by [@jdx](https://github.com/jdx) in [#559](#559) ### New Contributors - @mustafa0x made their first contribution in [#548](#548)
⚠️ **CAUTION: this is a major update, indicating a breaking change!**⚠️ This MR contains the following updates: | Package | Update | Change | |---|---|---| | [usage](https://github.com/jdx/usage) | major | `2.18.2` → `3.2.0` | MR created with the help of [el-capitano/tools/renovate-bot](https://gitlab.com/el-capitano/tools/renovate-bot). **Proposed changes to behavior should be submitted there as MRs.** --- ### Release Notes <details> <summary>jdx/usage (usage)</summary> ### [`v3.2.0`](https://github.com/jdx/usage/blob/HEAD/CHANGELOG.md#320---2026-03-23) [Compare Source](jdx/usage@v3.1.0...v3.2.0) ##### 🚀 Features - Support env-backed choices with `choices env=...` by [@​mustafa0x](https://github.com/mustafa0x) in [#​548](jdx/usage#548) ##### 🐛 Bug Fixes - **(zsh)** escape parentheses and brackets in completion descriptions by [@​jdx](https://github.com/jdx) in [#​559](jdx/usage#559) ##### New Contributors - [@​mustafa0x](https://github.com/mustafa0x) made their first contribution in [#​548](jdx/usage#548) ### [`v3.1.0`](https://github.com/jdx/usage/blob/HEAD/CHANGELOG.md#310---2026-03-22) [Compare Source](jdx/usage@v3.0.0...v3.1.0) ##### 🚀 Features - **(cli)** render all doc-related fields in --help output by [@​jdx](https://github.com/jdx) in [#​554](jdx/usage#554) - **(cli)** support reading spec from stdin via --file - by [@​jdx](https://github.com/jdx) in [#​555](jdx/usage#555) ##### 🐛 Bug Fixes - **(zsh)** remove trailing space from completions and add directory slash by [@​jdx](https://github.com/jdx) in [#​556](jdx/usage#556) - use field assignment for non-exhaustive Spec in benchmarks by [@​jdx](https://github.com/jdx) in [#​552](jdx/usage#552) ##### 📦️ Dependency Updates - update apple-actions/import-codesign-certs digest to [`fe74d46`](jdx/usage@fe74d46) by [@​renovate\[bot\]](https://github.com/renovate\[bot]) in [#​550](jdx/usage#550) - update codecov/codecov-action digest to [`1af5884`](jdx/usage@1af5884) by [@​renovate\[bot\]](https://github.com/renovate\[bot]) in [#​551](jdx/usage#551) - lock file maintenance by [@​renovate\[bot\]](https://github.com/renovate\[bot]) in [#​547](jdx/usage#547) ### [`v3.0.0`](https://github.com/jdx/usage/blob/HEAD/CHANGELOG.md#300---2026-03-13) [Compare Source](jdx/usage@v2.18.2...v3.0.0) ##### 🚀 Features - **(spec)** **breaking** add support for license, before/after help metadata by [@​jdx](https://github.com/jdx) in [#​542](jdx/usage#542) ##### 🐛 Bug Fixes - **(cobra)** escape newlines, tabs, and carriage returns in kdlQuoteAlways by [@​thecodesmith](https://github.com/thecodesmith) in [#​539](jdx/usage#539) - bump major version for breaking changes in release automation by [@​jdx](https://github.com/jdx) in [#​544](jdx/usage#544) - add custom\_major\_increment\_regex for breaking change detection by [@​jdx](https://github.com/jdx) in [#​545](jdx/usage#545) - handle all breaking change commit formats in major bump regex by [@​jdx](https://github.com/jdx) in [27e1ab1](jdx/usage@27e1ab1) - normalize breaking change commit format in preprocessor by [@​jdx](https://github.com/jdx) in [aa72b92](jdx/usage@aa72b92) ##### 📚 Documentation - add argparse-usage integration by [@​jdx](https://github.com/jdx) in [#​531](jdx/usage#531) - mark KDL code blocks as KDL and use correct inline-comment `//` by [@​muzimuzhi](https://github.com/muzimuzhi) in [#​536](jdx/usage#536) - fix include syntax to match implementation by [@​jdx](https://github.com/jdx) in [#​540](jdx/usage#540) - consolidate integration list to single source by [@​jdx](https://github.com/jdx) in [#​541](jdx/usage#541) - fix link to integrations by [@​muzimuzhi](https://github.com/muzimuzhi) in [#​543](jdx/usage#543) ##### 🛡️ Security - **(deps)** update dependency eslint to v10 by [@​renovate\[bot\]](https://github.com/renovate\[bot]) in [#​526](jdx/usage#526) ##### 🔍 Other Changes - Added an integration with ruby's OptionParser by [@​packrat386](https://github.com/packrat386) in [#​533](jdx/usage#533) ##### 📦️ Dependency Updates - update actions/setup-node digest to [`53b8394`](jdx/usage@53b8394) by [@​renovate\[bot\]](https://github.com/renovate\[bot]) in [#​525](jdx/usage#525) - update jdx/mise-action action to v3 by [@​renovate\[bot\]](https://github.com/renovate\[bot]) in [#​528](jdx/usage#528) - update rust crate roff to v1 by [@​renovate\[bot\]](https://github.com/renovate\[bot]) in [#​529](jdx/usage#529) ##### New Contributors - [@​thecodesmith](https://github.com/thecodesmith) made their first contribution in [#​539](jdx/usage#539) - [@​packrat386](https://github.com/packrat386) made their first contribution in [#​533](jdx/usage#533) </details> --- ### Configuration 📅 **Schedule**: Branch creation - At any time (no schedule defined), Automerge - At any time (no schedule defined). 🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied. ♻ **Rebasing**: Whenever MR becomes conflicted, or you tick the rebase/retry checkbox. 🔕 **Ignore**: Close this MR and you won't be reminded about this update again. --- - [ ] <!-- rebase-check -->If you want to rebase/retry this MR, check this box --- This MR has been generated by [Renovate Bot](https://github.com/renovatebot/renovate). <!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiI0My4xMDIuMTAiLCJ1cGRhdGVkSW5WZXIiOiI0My4xMDIuMTAiLCJ0YXJnZXRCcmFuY2giOiJtYWluIiwibGFiZWxzIjpbIlJlbm92YXRlIEJvdCIsImF1dG9tYXRpb246Ym90LWF1dGhvcmVkIiwiZGVwZW5kZW5jeS10eXBlOjptYWpvciJdfQ==-->
Summary
(,),[,]in zsh completion output to prevent_describefrom interpreting them as glob qualifiers/character classeszsh_escape()helper that handles all zsh-special characters (colons, parens, brackets)Fixes #558
Test plan
complete_word_zsh_escapes_parens_and_bracketsverifies escaped output🤖 Generated with Claude Code
Note
Low Risk
Low risk: only adjusts zsh completion output formatting by adding additional escaping, with coverage via a new integration test.
Overview
Fixes zsh completion output for
_describeby escaping additional special characters inname:descriptionentries. The previous inline colon escaping is replaced with a sharedzsh_escape()helper that also escapes(,),[and]to prevent glob/character-class interpretation.Adds an integration test plus a new example spec to verify zsh output is correctly escaped for descriptions containing parentheses/brackets (and colons).
Written by Cursor Bugbot for commit f165847. This will update automatically on new commits. Configure here.