Merge branch 'master' into intro-doc

This commit is contained in:
Matthew Dutson 2021-01-12 17:31:28 -06:00
commit d079026ecc
81 changed files with 1954 additions and 695 deletions

View file

@ -1,5 +1,5 @@
--- ---
Checks: 'clang-diagnostic-*,clang-analyzer-*,cert-*,performance-*,portability-*,modernize-use-auto,modernize-loop-convert,modernize-use-bool-literals,modernize-use-using,hicpp-uppercase-literal-suffix,readability-make-member-function-const,readability-redundant-string-init,readability-inconsistent-declaration-parameter-name,readability-redundant-access-specifiers,-performance-noexcept-move-constructor,-cert-dcl37-c,-cert-dcl51-cpp' Checks: 'clang-diagnostic-*,clang-analyzer-*,-clang-analyzer-valist.Uninitialized,cert-*,performance-*,portability-*,-modernize-use-auto,modernize-loop-convert,modernize-use-bool-literals,modernize-use-using,hicpp-uppercase-literal-suffix,readability-make-member-function-const,readability-redundant-string-init,readability-inconsistent-declaration-parameter-name,readability-redundant-access-specifiers,-performance-noexcept-move-constructor,-cert-dcl37-c,-cert-dcl50-cpp,-cert-dcl51-cpp,-cert-str34-c,-cert-env33-c'
WarningsAsErrors: '' WarningsAsErrors: ''
HeaderFilterRegex: '' HeaderFilterRegex: ''
AnalyzeTemporaryDtors: false AnalyzeTemporaryDtors: false

View file

@ -2,7 +2,7 @@ name: 'Lock threads'
on: on:
schedule: schedule:
- cron: '0 * * * *' - cron: '0 18 * * *'
jobs: jobs:
lock: lock:
@ -11,6 +11,6 @@ jobs:
- uses: dessant/lock-threads@v2 - uses: dessant/lock-threads@v2
with: with:
github-token: ${{ github.token }} github-token: ${{ github.token }}
issue-lock-inactive-days: '90' issue-lock-inactive-days: '180'
pr-lock-inactive-days: '90' pr-lock-inactive-days: '180'
issue-exclude-labels: 'question' issue-exclude-labels: 'question'

View file

@ -11,7 +11,7 @@ Notable improvements and fixes
# Show all dmesg lines related to "usb" # Show all dmesg lines related to "usb"
dmesg -w | string match '*usb*' dmesg -w | string match '*usb*'
- Prompts whose width exceeds $COLUMNS will now be truncated instead of replaced with `"> "` (:issue:`904`). - Prompts whose width exceeds $COLUMNS will now be truncated instead of replaced with ``"> "`` (:issue:`904`).
- When pressing Tab, fish displays ambiguous completions even when they - When pressing Tab, fish displays ambiguous completions even when they
have a common prefix, without the user having to press Tab again have a common prefix, without the user having to press Tab again
(:issue:`6924`). (:issue:`6924`).
@ -32,19 +32,20 @@ Notable improvements and fixes
1 = 2 and echo true or echo false 1 = 2 and echo true or echo false
^ ^
- The documentation (:issue:`6500`, :issue:`7371`) and ``fish_config`` (:issue:`7523`) received a new theme, matching the design on fishshell.com. This includes numbering the index from 1 instead of 0.
- The documentation (:issue:`6500`, :issue:`7371`) and Web-based configuration (:issue:`7523`) received a new theme, matching the design on fishshell.com.
- ``fish --no-execute`` will no longer complain about unknown commands - ``fish --no-execute`` will no longer complain about unknown commands
or non-matching wildcards, as these could be defined differently at or non-matching wildcards, as these could be defined differently at
runtime (especially for functions). This makes it usable as a static syntax checker (:issue:`977`). runtime (especially for functions). This makes it usable as a static syntax checker (:issue:`977`).
- ``type`` is now a builtin and therefore much faster (:issue:`7342`). - ``string match --regex`` now integrates named PCRE2 capture groups with fish variables, allowing variables to be set directly from ``string match`` (:issue:`7459`). To support this functionality, ``string`` is now a reserved word and can no longer be wrapped in a function.
- ``string match --regex`` now imports named PCRE2 capture groups as fish variables (:issue:`7459`). Note: Because of this, it can no longer be wrapped in a function and the name has been added as a reserved word. - Globs and other expansions are limited to 512,288 results (:issue:`7226`). Because operating systems limit arguments to ARG_MAX, larger values are unlikely to work anyway, and this helps to avoid hangs.
- Globs and other expansions are limited to 512k results (:issue:`7226`). Because operating systems limit arguments to ARG_MAX, larger values are unlikely to work anyway, and this helps to avoid hangs. - fish will now always attempt to become process group leader in interactive mode (:issue:`7060`). This helps avoid hangs in certain circumstances, and allows tmux's current directory introspection to work (:issue:`5699`).
- fish will now always attempt to become process group leader in interactive mode (:issue:`7060`). This helps avoid hangs in certain circumstances, and allows tmux' cwd-introspection hack to work (:issue:`5699`).
Syntax changes and new commands Syntax changes and new commands
------------------------------- -------------------------------
- Range limits in index range expansions like ``$x[$start..$end]`` may be omitted: ``$start`` and ``$end`` default to 1 and -1 (the last item) respectively. - Range limits in index range expansions like ``$x[$start..$end]`` may be omitted: ``$start`` and ``$end`` default to 1 and -1 (the last item) respectively.
- Logical operators ``&&`` and ``||`` can be followed by newlines before their right operand, matching POSIX shells. - Logical operators ``&&`` and ``||`` can be followed by newlines before their right operand, matching POSIX shells.
- When globbing, a segment which is exactly ``**`` may now match zero directories. For example ``**/foo`` may match ``foo`` in the current directory (:issue:`7222`).
Scripting improvements Scripting improvements
---------------------- ----------------------
@ -61,9 +62,11 @@ Scripting improvements
- The ``true`` and ``false`` builtins ignore any arguments, like other shells (:issue:`7030`). - The ``true`` and ``false`` builtins ignore any arguments, like other shells (:issue:`7030`).
- Computed ("electric") variables such as ``status`` are now only global in scope, so ``set -Uq status`` returns false (:issue:`7032`). - Computed ("electric") variables such as ``status`` are now only global in scope, so ``set -Uq status`` returns false (:issue:`7032`).
- The output for ``set --show`` has been shortened, only mentioning the scopes in which a variable exists (:issue:`6944`). - The output for ``set --show`` has been shortened, only mentioning the scopes in which a variable exists (:issue:`6944`).
In addition it now shows if a variable is a path variable.
- A new ``fish_posterror`` event is emitted when attempting to execute a command with syntax errors (:issue:`6880`). - A new ``fish_posterror`` event is emitted when attempting to execute a command with syntax errors (:issue:`6880`).
- ``fish_indent`` now removes spurious quotes in simple cases (:issue:`6722`) - ``fish_indent`` now removes unnecessary quotes in simple cases (:issue:`6722`)
and learned a ``--check`` option to just check if a file is indented correctly (:issue:`7251`). and learned a ``--check`` option to just check if a file is indented correctly (:issue:`7251`).
- ``fish_indent`` indents continuation lines that follow a line ending in a backslash, ``|``, ``&&`` or ``||``.
- ``pushd`` only adds a directory to the stack if changing to it was successful (:issue:`6947`). - ``pushd`` only adds a directory to the stack if changing to it was successful (:issue:`6947`).
- A new ``fish_job_summary`` function is called whenever a - A new ``fish_job_summary`` function is called whenever a
background job stops or ends, or any job terminates from a signal (:issue:`6959`). background job stops or ends, or any job terminates from a signal (:issue:`6959`).
@ -74,14 +77,16 @@ Scripting improvements
- ``status`` gained new ``dirname`` and ``basename`` convenience subcommands - ``status`` gained new ``dirname`` and ``basename`` convenience subcommands
to get just the directory to the running script or the name of it, to get just the directory to the running script or the name of it,
to simplify common tasks such as running ``(dirname (status filename))`` (:issue:`7076`). to simplify common tasks such as running ``(dirname (status filename))`` (:issue:`7076`).
- The ``_`` gettext function is now implemented as a builtin for performance purposes (:issue:`7036`). - The ``type`` and ``_`` gettext functions are now implemented as a builtin for performance purposes (:issue:`7342`, :issue:`7036`).
- Broken pipelines are now handled more smoothly; in particular, bad redirection mid-pipeline - Broken pipelines are now handled more smoothly; in particular, bad redirection mid-pipeline
results in the job continuing to run but with the broken file descriptor replaced with a closed results in the job continuing to run but with the broken file descriptor replaced with a closed
file descriptor. This allows better error recovery and is more in line with other shells' file descriptor. This allows better error recovery and is more in line with other shells'
behaviour (:issue:`7038`). behaviour (:issue:`7038`).
- ``jobs --quiet PID`` no longer prints "no suitable job" if the job for PID does not exist (eg because it has finished) (:issue:`6809`). - ``jobs --quiet PID`` no longer prints "no suitable job" if the job for PID does not exist (eg because it has finished) (:issue:`6809`).
- All builtins that query if something exists now take ``--query`` as the long form for ``-q``. ``--quiet`` is deprecated for ``command``, ``jobs`` and ``type`` (:issue:`7276`). - ``command``, ``jobs`` and ``type`` builtins support ``--query`` as the long form of ``-q``, matching other builtins. The long form ``--quiet`` is deprecated (:issue:`7276`).
- ``argparse`` now only prints a backtrace with invalid options to argparse itself (:issue:`6703`). - ``argparse`` no longer requires a short flag letter for long-only options (:issue:`7585`) and only prints a backtrace with invalid options to argparse itself (:issue:`6703`).
- ``argparse`` now passes the validation variables (e.g. ``$_flag_value``) as local-exported variables,
avoiding the need for ``--no-scope-shadowing`` in validation functions.
- ``complete`` takes the first argument as the name of the command if the ``--command``/``-c`` option is not used (``complete git`` is treated like ``complete --command git``), and can show the loaded completions for specific commands with ``complete COMMANDNAME`` (:issue:`7321`). - ``complete`` takes the first argument as the name of the command if the ``--command``/``-c`` option is not used (``complete git`` is treated like ``complete --command git``), and can show the loaded completions for specific commands with ``complete COMMANDNAME`` (:issue:`7321`).
- ``set_color -b`` (without an argument) no longer prints an error message, matching other invalid invocations of this command (:issue:`7154`). - ``set_color -b`` (without an argument) no longer prints an error message, matching other invalid invocations of this command (:issue:`7154`).
- Functions triggered by the ``fish_exit`` event are correctly run when the terminal is closed or the shell receives SIGHUP (:issue:`7014`). - Functions triggered by the ``fish_exit`` event are correctly run when the terminal is closed or the shell receives SIGHUP (:issue:`7014`).
@ -92,11 +97,23 @@ Scripting improvements
- ``set --erase`` and ``abbr --erase`` can now erase multiple things in one go, matching ``functions --erase`` (:issue:`7377`). - ``set --erase`` and ``abbr --erase`` can now erase multiple things in one go, matching ``functions --erase`` (:issue:`7377`).
- ``abbr --erase`` no longer errors on an unset abbreviation (:issue:`7376`). - ``abbr --erase`` no longer errors on an unset abbreviation (:issue:`7376`).
- ``test -t``, for testing whether file descriptors are connected to a terminal, works for file descriptors 0, 1, and 2 (:issue:`4766`). It can still return incorrect results in other cases (:issue:`1228`). - ``test -t``, for testing whether file descriptors are connected to a terminal, works for file descriptors 0, 1, and 2 (:issue:`4766`). It can still return incorrect results in other cases (:issue:`1228`).
- Trying to run scripts with Windows line endings (CRLF) via the shebang produces a sensible error (:issue:`2783`). - Trying to execute scripts with Windows line endings (CRLF) produces a sensible error (:issue:`2783`).
- An ``alias`` that delegates to a command with the same name no longer triggers an error about recursive completion (:issue:`7389`). - An ``alias`` that delegates to a command with the same name no longer triggers an error about recursive completion (:issue:`7389`).
- ``math`` now has a ``--base`` option to output the result in hexadecimal or octal (:issue:`7496`). - ``math`` now has a ``--base`` option to output the result in hexadecimal or octal (:issue:`7496`) and produces more specific error messages (:issue:`7508`).
- ``math`` learned bitwise functions ``bitand``, ``bitor`` and ``bitxor``, used like ``math "bitand(0xFE, 5)"`` (:issue:`7281`).
- ``math`` learned tau for those wishing to cut down on typing "2 * pi".
- ``string`` subcommands now quit early when used with ``--quiet`` (:issue:`7495`). - ``string`` subcommands now quit early when used with ``--quiet`` (:issue:`7495`).
- ``string repeat`` now handles multiple arguments, repeating each one (:issue:`5988`).
- Failed redirections will now set ``$status`` (:issue:`7540`). - Failed redirections will now set ``$status`` (:issue:`7540`).
- More consistent $status after errors, including invalid expansions like ``$foo[``.
- ``read`` can now read interactively from other files, so e.g. forcing it to read from the terminal via ``read </dev/tty`` works (:issue:`7358`).
- A new ``fish_status_to_signal`` function for transforming exit statuses to signal names (:issue:`7597`).
- The fallback ``realpath`` builtin supports the ``-s``/``--no-symlinks`` option, like GNU realpath.
- ``.`` and ``:`` are now also builtins instead of functions (:issue:`6854`).
- ``functions`` now explains when a function was defined via ``source`` instead of just saying ``Defined in -``.
- Significant performance improvements when globbing or in ``math``.
- ``echo`` no longer interprets options at the beginning of an argument (``echo "-n foo"``) (:issue:`7614`).
- Fish now better handles an unset $HOME (:issue:`7620`).
Interactive improvements Interactive improvements
------------------------ ------------------------
@ -113,7 +130,7 @@ Interactive improvements
- A new variable ``$status_generation`` is incremented only when the previous command produces a status (:issue:`6815`). This can be used, for example, to check whether a failure status is a holdover due to a background job, or actually produced by the last run command. - A new variable ``$status_generation`` is incremented only when the previous command produces a status (:issue:`6815`). This can be used, for example, to check whether a failure status is a holdover due to a background job, or actually produced by the last run command.
- ``fish_greeting`` is now a function that reads a variable of the same name, and defaults to setting it globally. This removes a universal variable by default and helps with updating the greeting. However, to disable the greeting it is now necessary to explicitly specify universal scope (``set -U fish_greeting``) or to disable it in config.fish (:issue:`7265`). - ``fish_greeting`` is now a function that reads a variable of the same name, and defaults to setting it globally. This removes a universal variable by default and helps with updating the greeting. However, to disable the greeting it is now necessary to explicitly specify universal scope (``set -U fish_greeting``) or to disable it in config.fish (:issue:`7265`).
- Events are properly emitted after a job is cancelled (:issue:`2356`). - Events are properly emitted after a job is cancelled (:issue:`2356`).
- A number of new debugging categories have been added, including ``config``, ``path``, ``reader`` and ``screen`` (:issue:`6511`). See the output of ``fish --print-debug-categories`` for the full list. - A number of new debugging categories have been added, including ``config``, ``path``, ``reader`` and ``screen`` (:issue:`6511`). See the output of ``fish --print-debug-categories`` for the full list. The old numbered debugging levels have been removed.
- The enabled debug categories are now printed on shell startup (:issue:`7007`). - The enabled debug categories are now printed on shell startup (:issue:`7007`).
- The ``-o`` short option to fish, for ``--debug-output``, works correctly instead of producing an - The ``-o`` short option to fish, for ``--debug-output``, works correctly instead of producing an
invalid option error (:issue:`7254`). invalid option error (:issue:`7254`).
@ -124,8 +141,8 @@ Interactive improvements
revealed. revealed.
- The output of ``time`` is now properly aligned in all cases (:issue:`6726`). - The output of ``time`` is now properly aligned in all cases (:issue:`6726`).
- The ``pwd`` command supports the long options ``--logical`` and ``--physical``, matching other implementations (:issue:`6787`). - The ``pwd`` command supports the long options ``--logical`` and ``--physical``, matching other implementations (:issue:`6787`).
- The command-not-found handling has been simplified. When it can't find a command, fish now just executes a function called ``fish_command_not_found`` instead of firing an event, making it easier to replace and reason about. Shims for backwards-compatibility have been added (:issue:`7293`). - The command-not-found handling has been simplified. When it can't find a command, fish now just executes a function called ``fish_command_not_found`` instead of firing an event, making it easier to replace and reason about. Previously-defined ``__fish_command_not_found_handler`` functions with an appropriate event listener will still work (:issue:`7293`).
- Control-C no longer occasionally prints an "unknown command" error (:issue:`7145`). - Control-C no longer occasionally prints an "unknown command" error (:issue:`7145`) or overwrites multiline prompts (:issue:`3537`).
- Autocompletions work properly after Control-C to cancel the commmand line (:issue:`6937`). - Autocompletions work properly after Control-C to cancel the commmand line (:issue:`6937`).
- History search is now case-insensitive unless the search string contains an uppercase character (:issue:`7273`). - History search is now case-insensitive unless the search string contains an uppercase character (:issue:`7273`).
- ``fish_update_completions`` has a new ``-keep`` option, which improves speed by skipping completions that already exist (:issue:`6775`). - ``fish_update_completions`` has a new ``-keep`` option, which improves speed by skipping completions that already exist (:issue:`6775`).
@ -134,28 +151,38 @@ Interactive improvements
- Long command lines no longer add a blank line after execution (:issue:`6826`) and behave better with backspace (:issue:`6951`). - Long command lines no longer add a blank line after execution (:issue:`6826`) and behave better with backspace (:issue:`6951`).
- ``functions -t`` works like the long option ``--handlers-type``, as documented, instead of producing an error (:issue:`6985`). - ``functions -t`` works like the long option ``--handlers-type``, as documented, instead of producing an error (:issue:`6985`).
- History search now flashes when it found no more results (:issue:`7362`) - History search now flashes when it found no more results (:issue:`7362`)
- Fish's debugging can now also be enabled via $FISH_DEBUG and $FISH_DEBUG_OUTPUT from the outside. This helps with debugging when no commandline options can be passed, like when fish is called in a shebang (:issue:`7359`). - fish's debugging can now also be enabled via $FISH_DEBUG and $FISH_DEBUG_OUTPUT from the outside. This helps with debugging when no commandline options can be passed, like when fish is called in a shebang (:issue:`7359`).
- Fish now creates $XDG_RUNTIME_DIR if it does not exist (:issue:`7335`). - fish now creates the path in the environment variable ``XDG_RUNTIME_DIR`` if it does not exist, before using it for runtime data storage (:issue:`7335`).
- ``set_color --print-colors`` now also respects the bold, dim, underline, reverse, italic and background modifiers, to better show their effect (:issue:`7314`). - ``set_color --print-colors`` now also respects the bold, dim, underline, reverse, italic and background modifiers, to better show their effect (:issue:`7314`).
- The fish Web configuration tool (``fish_config``) shows prompts correctly on Termux for Android (:issue:`7298`) and detects Windows Services for Linux 2 properly (:issue:`7027`). - The fish Web configuration tool (``fish_config``) shows prompts correctly on Termux for Android (:issue:`7298`) and detects Windows Services for Linux 2 properly (:issue:`7027`).
- ``funcsave`` has a new ``--directory`` option to specify the location of the saved function (:issue:`7041`). - ``funcsave`` has a new ``--directory`` option to specify the location of the saved function (:issue:`7041`).
- ``help`` works properly on MSYS2 (:issue:`7113`). - ``help`` works properly on MSYS2 (:issue:`7113`).
- Resuming a piped job by its number, like ``fg %1`` has been fixed (:issue:`7406`). - Resuming a piped job by its number, like ``fg %1``, works correctly (:issue:`7406`). Resumed jobs show the correct title in the terminal emulator (:issue:`7444`).
- Commands run from key bindings now use the same tty modes as normal commands (:issue:`7483`). - Commands run from key bindings now use the same TTY modes as normal commands (:issue:`7483`).
- Autosuggestions from history are now case-sensitive, and tab completions are "smartcase": they offer case-insensitive matches if the input string is lowercase (:issue:`3978`). - Autosuggestions from history are now case-sensitive, and tab completions are "smartcase": they offer case-insensitive matches if the input string is lowercase (:issue:`3978`).
- ``$status`` from completion scripts is no longer passed outside the completion, which keeps the status display in the prompt as the last command's status (:issue:`7555`).
- Updated localisations for pt_BR (:issue:`7480`).
- ``fish_trace`` output now starts with ``->`` like ``fish --profile``'s, making the depth more visible (:issue:`7538`).
- Resizing the terminal window no longer produces a corrupted prompt (:issue:`6532`).
- ``functions`` produces an error rather than crashing on certain invalid arguments (:issue:`7515`).
- A crash in using tab completions with inline variable assignment (eg ``A= b``) has been fixed (:issue:`7344`).
- ``fish_private_mode`` may now be changed dynamically using ``set`` (:issue:`7589`).
- Commands with leading spaces may be retrieved from history with up-arrow until a new command is run, matching zsh's ``HIST_IGNORE_SPACE`` (:issue:`1383`).
- Importing bash history or reporting errors with recursive globs (``**``) no longer hangs (:issue:`7407`, :issue:`7497`).
New or improved bindings New or improved bindings
^^^^^^^^^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^^^^^^
- As mentioned above, new readline commands ``undo`` (Control+\_ or Control+Z) and ``redo`` (Alt-/) can be used to revert changes to the command line or the pager search field (:issue:`6570`). - As mentioned above, new special input functions ``undo`` (Control+\_ or Control+Z) and ``redo`` (Alt-/) can be used to revert changes to the command line or the pager search field (:issue:`6570`).
- Control-Z is now available for binding (:issue:`7152`). - Control-Z is now available for binding (:issue:`7152`).
- Additionally, using the ``cancel`` readline command (bound to escape by default) right after fish picked an unambiguous completion will undo that (:issue:`7433`). - Additionally, using the ``cancel`` special input function (bound to escape by default) right after fish picked an unambiguous completion will undo that (:issue:`7433`).
- Vi mode bindings now support ``dh``, ``dl``, ``c0``, ``cf``, ``ct``, ``cF``, ``cT``, ``ch``, ``cl``, ``y0``, ``ci``, ``ca``, ``yi``, ``ya``, ``di``, ``da``, ``o``, ``O`` and Control+left/right keys to navigate by word (:issue:`6648`, :issue:`6755`, :issue:`6769`, :issue:`7442`). - Vi mode bindings now support ``dh``, ``dl``, ``c0``, ``cf``, ``ct``, ``cF``, ``cT``, ``ch``, ``cl``, ``y0``, ``ci``, ``ca``, ``yi``, ``ya``, ``di``, ``da``, ``d;``, ``d,``, ``o``, ``O`` and Control+left/right keys to navigate by word (:issue:`6648`, :issue:`6755`, :issue:`6769`, :issue:`7442`, :issue:`7516`).
- Vi mode bindings support ``~`` (tilde) to toggle the case of the selected character (:issue:`6908`). - Vi mode bindings support ``~`` (tilde) to toggle the case of the selected character (:issue:`6908`).
- Functions ``up-or-search`` and ``down-or-search`` (up-arrow and down-arrow) can cross empty lines and don't activate search mode if the search fails which makes it easier to use them to move between lines in some situations. - Functions ``up-or-search`` and ``down-or-search`` (up-arrow and down-arrow) can cross empty lines and don't activate search mode if the search fails which makes it easier to use them to move between lines in some situations.
- The readline command ``beginning-of-history`` (Page Up) now moves to the oldest search instead of the youngest - that's ``end-of-history`` (Page Down). - If history search fails to find a match, the cursor is no longer moved. This is useful when accidentally starting a history search on a multi-line commandline.
- A new readline command ``forward-single-char`` moves one character to the right, and if an autosuggestion is available, only take a single character from it (:issue:`7217`). - The special input function ``beginning-of-history`` (Page Up) now moves to the oldest search instead of the youngest - that's ``end-of-history`` (Page Down).
- Readline commands can now be joined with ``or`` as a modifier (adding to ``and``), though only some commands report success or failure (:issue:`7217`). - A new special input function ``forward-single-char`` moves one character to the right, and if an autosuggestion is available, only take a single character from it (:issue:`7217`).
- Special input functions can now be joined with ``or`` as a modifier (adding to ``and``), though only some commands set an exit status (:issue:`7217`). This includes ``suppress-autosuggestion`` to reflect whether an autosuggestion was suppressed (:issue:`1419`)
- A new function ``__fish_preview_current_file``, bound to Alt+O, opens the - A new function ``__fish_preview_current_file``, bound to Alt+O, opens the
current file at the cursor in a pager (:issue:`6838`). current file at the cursor in a pager (:issue:`6838`).
- ``edit_command_buffer`` (Alt-E and Alt-V) passes the cursor position - ``edit_command_buffer`` (Alt-E and Alt-V) passes the cursor position
@ -163,11 +190,12 @@ New or improved bindings
- ``__fish_prepend_sudo`` (Alt-S) now toggles a ``sudo`` prefix (:issue:`7012`) and avoids shifting the cursor (:issue:`6542`). - ``__fish_prepend_sudo`` (Alt-S) now toggles a ``sudo`` prefix (:issue:`7012`) and avoids shifting the cursor (:issue:`6542`).
- ``__fish_prepend_sudo`` (Alt-S) now uses the previous commandline if the current one is empty, - ``__fish_prepend_sudo`` (Alt-S) now uses the previous commandline if the current one is empty,
to simplify rerunning the previous command with ``sudo`` (:issue:`7079`). to simplify rerunning the previous command with ``sudo`` (:issue:`7079`).
- ``__fish_toggle_comment_commandline`` (Alt-#) now uncomments and presents the last comment - ``__fish_toggle_comment_commandline`` (Alt-#) now uncomments and presents the last comment
from history if the commandline is empty (:issue:`7137`). from history if the commandline is empty (:issue:`7137`).
- ``__fish_whatis_current_token`` (Alt-W) prints descriptions for functions and builtins (:issue:`7191`). - ``__fish_whatis_current_token`` (Alt-W) prints descriptions for functions and builtins (:issue:`7191`).
- The definition of "word" and "bigword" for movements was refined, fixing e.g. vi mode's behavior with ``e`` on the second-to-last char, and bigword's behavior with single-char words and non-blank non-graphic characters (:issue:`7353`, :issue:`7354`, :issue:`4025`, :issue:`7328`, :issue:`7325`) - The definition of "word" and "bigword" for movements was refined, fixing (eg) vi mode's behavior with ``e`` on the second-to-last char, and bigword's behavior with single-char words and non-blank non-graphic characters (:issue:`7353`, :issue:`7354`, :issue:`4025`, :issue:`7328`, :issue:`7325`)
- fish's clipboard bindings now also support WSL via powershell and clip.exe (:issue:`7455`) and will properly copy newlines in multi-line commands.
- Using the ``*-jump`` special input functions before typing anything else no longer crashes fish.
Improved prompts Improved prompts
^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^
@ -176,20 +204,21 @@ Improved prompts
commands prefixed with ``not`` (:issue:`6566`). commands prefixed with ``not`` (:issue:`6566`).
- git prompts include all untracked files in the repository, not just those in the current - git prompts include all untracked files in the repository, not just those in the current
directory (:issue:`6086`). directory (:issue:`6086`).
- The git prompts correctly show stash states (:issue:`6876`, :issue:`7136`). - The git prompts correctly show stash states (:issue:`6876`, :issue:`7136`) and clean states (:issue:`7471`).
- The Mercurial prompt correctly shows untracked status (:issue:`6906`). - The Mercurial prompt correctly shows untracked status (:issue:`6906`), and by default only shows the branch for performance reasons.
A new variable ``$fish_prompt_hg_show_informative_status`` can be set to enable more information.
- The ``fish_vcs_prompt`` passes its arguments to the various VCS prompts that it calls (:issue:`7033`). - The ``fish_vcs_prompt`` passes its arguments to the various VCS prompts that it calls (:issue:`7033`).
- The Subversion prompt was broken in a number of ways in 3.1.0 and has been restored (:issue:`7278`). - The Subversion prompt was broken in a number of ways in 3.1.0 and has been restored (:issue:`6715`, :issue:`7278`).
- A new helper function ``fish_is_root_user`` simplifies checking for superuser privilege (:issue:`7031`). - A new helper function ``fish_is_root_user`` simplifies checking for superuser privilege (:issue:`7031`).
- New colorschemes - ``ayu Light``, ``ayu Dark`` and ``ayu Mirage`` (:issue:`7596`).
Improved terminal support Improved terminal support
^^^^^^^^^^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^^^^^^^
- A new variable, ``$fish_vi_force_cursor``, can be set to force ``fish_vi_cursor`` to attempt changing the cursor - A new variable, ``$fish_vi_force_cursor``, can be set to force ``fish_vi_cursor`` to attempt changing the cursor
shape in vi mode, regardless of terminal (:issue:`6968`). The ``fish_vi_cursor`` option ``--force-iterm`` has been deprecated. shape in vi mode, regardless of terminal (:issue:`6968`). The ``fish_vi_cursor`` option ``--force-iterm`` has been deprecated.
- ``diff`` will now colourise output, if supported (:issue:`7308`). - ``diff`` will now colourize output, if supported (:issue:`7308`).
- Autosuggestions now show up also when the cursor passes the right - Autosuggestions appear when the cursor passes the right prompt (:issue:`6948`) or wraps to the next line (:issue:`7213`).
prompt (:issue:`6948`) or wraps to the next line (:issue:`7213`).
- The cursor shape in Vi mode changes properly in Windows Terminal (:issue:`6999`). - The cursor shape in Vi mode changes properly in Windows Terminal (:issue:`6999`).
- The spurious warning about terminal size in small terminals has been removed (:issue:`6980`). - The spurious warning about terminal size in small terminals has been removed (:issue:`6980`).
- Dynamic titles are now enabled in Alacritty (:issue:`7073`). - Dynamic titles are now enabled in Alacritty (:issue:`7073`).
@ -198,7 +227,9 @@ Improved terminal support
- An issue producing strange status output from commands involving ``not`` has been fixed (:issue:`6566`). - An issue producing strange status output from commands involving ``not`` has been fixed (:issue:`6566`).
- Long command lines are wrapped in all cases, instead of sometimes being put on a new line (:issue:`5118`). - Long command lines are wrapped in all cases, instead of sometimes being put on a new line (:issue:`5118`).
- The pager is properly rendered with long command lines selected (:issue:`2557`). - The pager is properly rendered with long command lines selected (:issue:`2557`).
- Fish no longer performs its own resizing in VTE-based terminals, as they perform their own reflowing, which clashes especially with right prompts (:issue:`7491`). - Sessions with right prompts can be resized correctly in GNOME Terminal (and other VTE-based terminals) and Alacritty (:issue:`7491`).
- fish now sets terminal modes sooner, which stops output from appearing before the greeting and prompt are ready (:issue:`7489`).
- Better detection of new Konsole versions for truecolor support and cursor shape changing.
Completions Completions
^^^^^^^^^^^ ^^^^^^^^^^^
@ -209,7 +240,10 @@ Completions
- ``alias`` (:issue:`7035`) - ``alias`` (:issue:`7035`)
- ``apk`` (:issue:`7108`) - ``apk`` (:issue:`7108`)
- ``asciidoctor`` (:issue:`7000`) - ``asciidoctor`` (:issue:`7000`)
- ``bootctl`` (:issue:`7428`)
- ``bluetoothctl`` (:issue:`7438`)
- ``cmark`` (:issue:`7000`) - ``cmark`` (:issue:`7000`)
- ``coredumpctl`` (:issue:`7428`)
- ``create_ap`` (:issue:`7096`) - ``create_ap`` (:issue:`7096`)
- ``deno`` (:issue:`7138`) - ``deno`` (:issue:`7138`)
- ``dhclient`` - ``dhclient``
@ -221,9 +255,13 @@ Completions
- ``gh`` (:issue:`7112`) - ``gh`` (:issue:`7112`)
- ``gitk`` - ``gitk``
- ``hikari`` (:issue:`7083`) - ``hikari`` (:issue:`7083`)
- ``homectl`` (:issue:`7435`)
- ``hostnamectl`` (:issue:`7428`)
- ``icdiff`` (:issue:`7503`)
- ``imv`` (:issue:`6675`) - ``imv`` (:issue:`6675`)
- ``julia`` (:issue:`7468`) - ``julia`` (:issue:`7468`)
- ``k3d`` (:issue:`7202`) - ``k3d`` (:issue:`7202`)
- ``ldapsearch`` (:issue:`7578`)
- ``micro`` (:issue:`7339`) - ``micro`` (:issue:`7339`)
- ``mpc`` (:issue:`7169`) - ``mpc`` (:issue:`7169`)
- Metasploit's ``msfconsole``, ``msfdb`` and ``msfvenom`` (:issue:`6930`) - Metasploit's ``msfconsole``, ``msfdb`` and ``msfvenom`` (:issue:`6930`)
@ -249,15 +287,25 @@ Completions
- ``zopfli`` and ``zopflipng`` - ``zopfli`` and ``zopflipng``
- Lots of improvements to completions. - Lots of improvements to completions.
- Improvements to the manpage completion generator (:issue:`7086`). - Improvements to the manpage completion generator (:issue:`7086`, :issue:`6879`).
- Significant performance improvements to completion of the available commands (:issue:`7153`). - Significant performance improvements to completion of the available commands (:issue:`7153`), especially on macOS Big Sur where there was a significant regression (:issue:`7365`).
- ``__fish_complete_suffix`` now uses the same fuzzy matching logic as normal file completion.
- ``__fish_complete_suffix`` completes any file but sorts files with matching suffix first (:issue:`7040`). Previously, it only completed files with matching suffix. - ``__fish_complete_suffix`` completes any file but sorts files with matching suffix first (:issue:`7040`). Previously, it only completed files with matching suffix.
- Completions for ``git`` learned to complete the right and left parts of a commit range like ``from..to`` or ``left...right``.
- The ``__fish_print_packages`` function was broken apart into one function per package manager, and any completion now only calls its specific function. This helps if multiple package managers are installed on a system (e.g. to create containers). ``__fish_print_packages`` remains as a stub that calls all functions (:issue:`7542`).
- Many completions have their descriptions shortened to fit more options on the screen (:issue:`6981`, :issue:`7550`, :issue:`7109`, :issue:`7569`, :issue:`7081`, :issue:`7291`, :issue:`7163`, :issue:`7378`).
- The ``make`` completions no longer second-guess make's file detection, fixing target completion in some cases (:issue:`7535`).
- The command completions now correctly print the description even if the command was fully matched (like in ``ls<TAB>``).
- The ``set`` completions no longer hide variables starting with ``__``, they are sorted last instead.
Deprecations and removed features Deprecations and removed features
--------------------------------- ---------------------------------
- fish no longer attempts to modify the terminal size via `TIOCSWINSZ` (:issue:`6994`). - fish no longer attempts to modify the terminal size via ``TIOCSWINSZ`` (:issue:`6994`).
- The `fish_color_match` variable is no longer used. (Previously this controlled the color of matching quotes and parens when using `read`). - The ``fish_color_match`` variable is no longer used. (Previously this controlled the color of matching quotes and parens when using ``read``).
- fish 3.2.0 will be the last release in which the redirection to standard error with the ``^`` character is enabled. The ``stderr-nocaret`` feature flag will be changed to "on" in future releases. - fish 3.2.0 will be the last release in which the redirection to standard error with the ``^`` character is enabled. The ``stderr-nocaret`` feature flag will be changed to "on" in future releases.
- ``string`` is now a reserved word and cannot be used for function names (see above).
- ``fish_vi_cursor``'s option ``--force-iterm`` has been deprecated (see above).
- ``command``, ``jobs`` and ``type`` long-form option ``--quiet`` is deprecated in favor of ``--query`` (see above).
For distributors and developers For distributors and developers
------------------------------- -------------------------------
@ -275,7 +323,8 @@ For distributors and developers
codesigning is enabled (:issue:`6952`). codesigning is enabled (:issue:`6952`).
- Running the full interactive test suite now requires Python 3.5+ and the pexpect package (:issue:`6825`); the expect package is no longer required. - Running the full interactive test suite now requires Python 3.5+ and the pexpect package (:issue:`6825`); the expect package is no longer required.
- Support for Python 2 in fish's tools (``fish_config`` and the manual page completion generator) is no longer guaranteed. Please use Python 3.5 or later (:issue:`6537`). - Support for Python 2 in fish's tools (``fish_config`` and the manual page completion generator) is no longer guaranteed. Please use Python 3.5 or later (:issue:`6537`).
- The webconfig tool no longer requires python's distutils (:issue:`7514`) - The Web-based configuration tool is compatible with Python 3.10 (:issue:`7600`) and no longer requires Python's distutils package (:issue:`7514`).
- fish 3.2 is the last release to support Red Hat Enterprise Linux & CentOS version 6.
-------------- --------------
@ -530,8 +579,6 @@ Scripting improvements
- ``math`` reports the right error when incorrect syntax is used inside - ``math`` reports the right error when incorrect syntax is used inside
parentheses (:issue:`6063`), and warns when unsupported logical operations parentheses (:issue:`6063`), and warns when unsupported logical operations
are used (:issue:`6096`). are used (:issue:`6096`).
- ``math`` learned bitwise functions ``bitand``, ``bitor`` and ``bitxor``, used like ``math "bitand(0xFE, 5)"`` (:issue:`7281`).
- ``math`` learned tau for those wishing to cut down on typing "2 * pi".
- ``functions --erase`` now also prevents fish from autoloading a - ``functions --erase`` now also prevents fish from autoloading a
function for the first time (:issue:`5951`). function for the first time (:issue:`5951`).
- ``jobs --last`` returns 0 to indicate success when a job is found - ``jobs --last`` returns 0 to indicate success when a job is found
@ -2310,7 +2357,7 @@ Other Notable Fixes
- Tab completions now work properly within nested subcommands. :issue:`913` - Tab completions now work properly within nested subcommands. :issue:`913`
- ``printf`` supports `\e`, the escape character. :issue:`910` - ``printf`` supports ``\e``, the escape character. :issue:`910`
- ``fish_config history`` no longer shows duplicate items. :issue:`900` - ``fish_config history`` no longer shows duplicate items. :issue:`900`

View file

@ -128,11 +128,6 @@ If you use Vim I recommend the `vim-clang-format
plugin <https://github.com/rhysd/vim-clang-format>`__ by plugin <https://github.com/rhysd/vim-clang-format>`__ by
[@rhysd](https://github.com/rhysd). [@rhysd](https://github.com/rhysd).
You can also get Vim to provide reasonably correct behavior by
installing
http://www.vim.org/scripts/script.php?script_id=2636
Emacs Emacs
^^^^^ ^^^^^
@ -177,8 +172,8 @@ made to run fish_indent via e.g.
Suppressing Reformatting of C++ Code Suppressing Reformatting of C++ Code
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
If you have a good reason for doing so you can tell ``clang-format`` to You can tell ``clang-format`` to not reformat a block by enclosing it in
not reformat a block of code by enclosing it in comments like this: comments like this:
:: ::
@ -186,10 +181,6 @@ not reformat a block of code by enclosing it in comments like this:
code to ignore code to ignore
// clang-format on // clang-format on
However, as I write this there are no places in the code where we use
this and I cant think of any legitimate reasons for exempting blocks of
code from clang-format.
Fish Script Style Guide Fish Script Style Guide
----------------------- -----------------------
@ -358,7 +349,7 @@ To install the lint checkers on Debian-based Linux distributions:
sudo apt-get install oclint sudo apt-get install oclint
sudo apt-get install cppcheck sudo apt-get install cppcheck
Installing the Reformatting Tools Installing the Formatting Tools
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Mac OS X: Mac OS X:
@ -371,15 +362,7 @@ Debian-based:
:: ::
apt-cache search clang-format sudo apt-get install clang-format
Above will list all the versions available. Pick the newest one
available (3.9 for Ubuntu 16.10 as I write this) and install it:
::
sudo apt-get install clang-format-3.9
sudo ln -s /usr/bin/clang-format-3.9 /usr/bin/clang-format
Message Translations Message Translations
-------------------- --------------------

View file

@ -43,9 +43,8 @@ build/fish: build/$(BUILDFILE)
# Use build as an order-only dependency. This prevents the target from always being outdated # Use build as an order-only dependency. This prevents the target from always being outdated
# after a make run, and more importantly, doesn't clobber manually specified CMake options. # after a make run, and more importantly, doesn't clobber manually specified CMake options.
build/$(BUILDFILE): | build build/$(BUILDFILE): | build
cd build; $(CMAKE) .. -DCMAKE_EXPORT_COMPILE_COMMANDS=1 -G "$(GENERATOR)" \ cd build; $(CMAKE) .. -G "$(GENERATOR)" \
-DCMAKE_INSTALL_PREFIX="$(PREFIX)" -DCMAKE_EXPORT_COMPILE_COMMANDS=1 \ -DCMAKE_INSTALL_PREFIX="$(PREFIX)" -DCMAKE_EXPORT_COMPILE_COMMANDS=1
-DCMAKE_BUILD_TYPE=RelWithDebInfo
build: build:
mkdir -p build mkdir -p build

View file

@ -1,7 +1,7 @@
#!/bin/sh #!/bin/sh
cppcheck --enable=all --std=posix --quiet \ cppcheck --std=posix --quiet \
--suppressions-list=build_tools/cppcheck.suppressions \ --suppressions-list=build_tools/cppcheck.suppressions --inline-suppr \
--rule-file=build_tools/cppcheck.rules \ --rule-file=build_tools/cppcheck.rules \
--force \ --force \
./src/ ${@:---enable=all ./src/}

View file

@ -18,11 +18,11 @@ argparse a/all p/project= -- $argv
# We only want -D and -I options to be passed thru to cppcheck. # We only want -D and -I options to be passed thru to cppcheck.
for arg in $argv for arg in $argv
if string match -q -- '-D*' $arg if string match -q -- '-D*' $arg
set cppcheck_args $cppcheck_args $arg set -a cppcheck_args (string split -- ' ' $arg)
else if string match -q -- '-I*' $arg else if string match -q -- '-I*' $arg
set cppcheck_args $cppcheck_args $arg set -a cppcheck_args (string split -- ' ' $arg)
else if string match -q -- '-iquote*' $arg else if string match -q -- '-iquote*' $arg
set cppcheck_args $cppcheck_args $arg set -a cppcheck_args (string split -- ' ' $arg)
end end
end end
@ -83,20 +83,7 @@ if set -q c_files[1]
echo ======================================== echo ========================================
echo Running cppcheck echo Running cppcheck
echo ======================================== echo ========================================
# The stderr to stdout redirection is because cppcheck, incorrectly IMHO, writes its build_tools/cppcheck.sh --enable=$cppchecks $c_files 2>&1
# diagnostic messages to stderr. Anyone running this who wants to capture its output will
# expect those messages to be written to stdout.
set -l cn (set_color normal)
set -l cb (set_color --bold)
set -l cu (set_color --underline)
set -l cm (set_color magenta)
set -l cbrm (set_color brmagenta)
set -l template "[$cb$cu{file}$cn$cb:{line}$cn] $cbrm{severity}$cm ({id}):$cn\n {message}"
set cppcheck_args -q --verbose --std=c++11 --std=posix --language=c++ --template $template \
--suppress=missingIncludeSystem --inline-suppr --enable=$cppchecks \
--rule-file=.cppcheck.rules --suppressions-list=.cppcheck.suppressions $cppcheck_args
cppcheck $cppcheck_args $c_files 2>&1
echo echo
echo ======================================== echo ========================================

View file

@ -28,8 +28,8 @@ if test $all = yes
exit 1 exit 1
end end
set c_files src/*.h src/*.cpp src/*.c set c_files src/*.h src/*.cpp src/*.c
set fish_files (printf '%s\n' share/***.fish) set fish_files share/**.fish
set python_files **.py set python_files {doc_src,share,tests}/**.py
else else
# We haven't been asked to reformat all the source. If there are uncommitted changes reformat # We haven't been asked to reformat all the source. If there are uncommitted changes reformat
# those using `git clang-format`. Else reformat the files in the most recent commit. # those using `git clang-format`. Else reformat the files in the most recent commit.

View file

@ -18,9 +18,9 @@ This command makes it easy for fish scripts and functions to handle arguments li
Each option specification (``OPTION_SPEC``) is written in the `domain specific language <#option-specifications>`__ described below. All OPTION_SPECs must appear after any argparse flags and before the ``--`` that separates them from the arguments to be parsed. Each option specification (``OPTION_SPEC``) is written in the `domain specific language <#option-specifications>`__ described below. All OPTION_SPECs must appear after any argparse flags and before the ``--`` that separates them from the arguments to be parsed.
Each option that is seen in the ARG list will result in a var name of the form ``_flag_X``, where ``X`` is the short flag letter and the long flag name. The OPTION_SPEC always requires a short flag even if it can't be used. So there will always be ``_flag_X`` var set using the short flag letter if the corresponding short or long flag is seen. The long flag name var (e.g., ``_flag_help``) will only be defined, obviously, if the OPTION_SPEC includes a long flag name. Each option that is seen in the ARG list will result in variables named ``_flag_X``, where ``X`` is the short flag letter and the long flag name (if they are defined). For example a ``--help`` option could cause argparse to define one variable called ``_flag_h`` and another called ``_flag_help``.
For example ``_flag_h`` and ``_flag_help`` if ``-h`` or ``--help`` is seen. The var will be set with local scope (i.e., as if the script had done ``set -l _flag_X``). If the flag is a boolean (that is, it just is passed or not, it doesn't have a value) the values are the short and long flags seen. If the option is not a boolean the values will be zero or more values corresponding to the values collected when the ARG list is processed. If the flag was not seen the flag var will not be set. The variables will be set with local scope (i.e., as if the script had done ``set -l _flag_X``). If the flag is a boolean (that is, it just is passed or not, it doesn't have a value) the values are the short and long flags seen. If the option is not a boolean the values will be zero or more values corresponding to the values collected when the ARG list is processed. If the flag was not seen the flag variable will not be set.
Options Options
------- -------
@ -74,13 +74,11 @@ Option Specifications
Each option specification consists of: Each option specification consists of:
- A short flag letter (which is mandatory). It must be an alphanumeric or "#". The "#" character is special and means that a flag of the form ``-123`` is valid. The short flag "#" must be followed by "-" (since the short name isn't otherwise valid since ``_flag_#`` is not a valid var name) and must be followed by a long flag name with no modifiers. - An optional alphanumeric short flag letter, followed by a ``/`` if the short flag can be used by someone invoking your command or, for backwards compatibility, a ``-`` if it should not be exposed as a valid short flag (in which case it will also not be exposed as a flag variable).
- A ``/`` if the short flag can be used by someone invoking your command else ``-`` if it should not be exposed as a valid short flag. If there is no long flag name these characters should be omitted. You can also specify a '#' to indicate the short and long flag names can be used and the value can be specified as an implicit int; i.e., a flag of the form ``-NNN``. - An optional long flag name. If not present then only the short flag letter can be used, and if that is not present either it's an error.
- A long flag name which is optional. If not present then only the short flag letter can be used. - Nothing if the flag is a boolean that takes no argument or is an integer flag, or
- Nothing if the flag is a boolean that takes no argument or is an implicit int flag, or
- ``=`` if it requires a value and only the last instance of the flag is saved, or - ``=`` if it requires a value and only the last instance of the flag is saved, or
@ -94,6 +92,17 @@ See the :ref:`fish_opt <cmd-fish_opt>` command for a friendlier but more verbose
If a flag is not seen when parsing the arguments then the corresponding _flag_X var(s) will not be set. If a flag is not seen when parsing the arguments then the corresponding _flag_X var(s) will not be set.
Integer flag
------------
Sometimes commands take numbers directly as options, like ``foo -55``. To allow this one option spec can have the ``#`` modifier so that any integer will be understood as this flag, and the last number will be given as its value (as if ``=`` was used).
The ``#`` must follow the short flag letter (if any), and other modifiers like ``=`` are not allowed, except for ``-`` (for backwards compatibility)::
m#maximum
This does not read numbers given as ``+NNN``, only those that look like flags - ``-NNN``.
Note: Optional arguments Note: Optional arguments
------------------------ ------------------------
@ -149,6 +158,10 @@ Some OPTION_SPEC examples:
- ``h-help`` means that only ``--help`` is valid. The flag is a boolean and can be used more than once. If the long flag is used then ``_flag_h`` and ``_flag_help`` will be set to the count of how many times the long flag was seen. - ``h-help`` means that only ``--help`` is valid. The flag is a boolean and can be used more than once. If the long flag is used then ``_flag_h`` and ``_flag_help`` will be set to the count of how many times the long flag was seen.
- ``help`` means that only ``--help`` is valid and only ``_flag_help`` will be set.
- ``longonly=`` is a flag ``--longonly`` that requires an option, there is no short flag or even short flag variable.
- ``n/name=`` means that both ``-n`` and ``--name`` are valid. It requires a value and can be used at most once. If the flag is seen then ``_flag_n`` and ``_flag_name`` will be set with the single mandatory value associated with the flag. - ``n/name=`` means that both ``-n`` and ``--name`` are valid. It requires a value and can be used at most once. If the flag is seen then ``_flag_n`` and ``_flag_name`` will be set with the single mandatory value associated with the flag.
- ``n/name=?`` means that both ``-n`` and ``--name`` are valid. It accepts an optional value and can be used at most once. If the flag is seen then ``_flag_n`` and ``_flag_name`` will be set with the value associated with the flag if one was provided else it will be set with no values. - ``n/name=?`` means that both ``-n`` and ``--name`` are valid. It accepts an optional value and can be used at most once. If the flag is seen then ``_flag_n`` and ``_flag_name`` will be set with the value associated with the flag if one was provided else it will be set with no values.
@ -165,6 +178,8 @@ Some OPTION_SPEC examples:
- ``n#max`` means that flags matching the regex "^--?\\d+$" are valid. When seen they are assigned to the variables ``_flag_n`` and ``_flag_max``. This allows any valid positive or negative integer to be specified by prefixing it with a single "-". Many commands support this idiom. For example ``head -3 /a/file`` to emit only the first three lines of /a/file. You can also specify the value using either flag: ``-n NNN`` or ``--max NNN`` in this example. - ``n#max`` means that flags matching the regex "^--?\\d+$" are valid. When seen they are assigned to the variables ``_flag_n`` and ``_flag_max``. This allows any valid positive or negative integer to be specified by prefixing it with a single "-". Many commands support this idiom. For example ``head -3 /a/file`` to emit only the first three lines of /a/file. You can also specify the value using either flag: ``-n NNN`` or ``--max NNN`` in this example.
After parsing the arguments the ``argv`` var is set with local scope to any values not already consumed during flag processing. If there are not unbound values the var is set but ``count $argv`` will be zero. - ``#longonly`` causes the last integer option to be stored in ``_flag_longonly``.
After parsing the arguments the ``argv`` variable is set with local scope to any values not already consumed during flag processing. If there are no unbound values the variable is set but ``count $argv`` will be zero.
If an error occurs during argparse processing it will exit with a non-zero status and print error messages to stderr. If an error occurs during argparse processing it will exit with a non-zero status and print error messages to stderr.

View file

@ -28,6 +28,8 @@ The generic key binding that matches if no other binding does can be set by spec
If the ``-k`` switch is used, the name of a key (such as 'down', 'up' or 'backspace') is used instead of a sequence. The names used are the same as the corresponding curses variables, but without the 'key\_' prefix. (See ``terminfo(5)`` for more information, or use ``bind --key-names`` for a list of all available named keys). Normally this will print an error if the current ``$TERM`` entry doesn't have a given key, unless the ``-s`` switch is given. If the ``-k`` switch is used, the name of a key (such as 'down', 'up' or 'backspace') is used instead of a sequence. The names used are the same as the corresponding curses variables, but without the 'key\_' prefix. (See ``terminfo(5)`` for more information, or use ``bind --key-names`` for a list of all available named keys). Normally this will print an error if the current ``$TERM`` entry doesn't have a given key, unless the ``-s`` switch is given.
To find out what sequence a key combination sends, you can use :ref:`fish_key_reader <cmd-fish_key_reader>`.
``COMMAND`` can be any fish command, but it can also be one of a set of special input functions. These include functions for moving the cursor, operating on the kill-ring, performing tab completion, etc. Use ``bind --function-names`` for a complete list of these input functions. ``COMMAND`` can be any fish command, but it can also be one of a set of special input functions. These include functions for moving the cursor, operating on the kill-ring, performing tab completion, etc. Use ``bind --function-names`` for a complete list of these input functions.
When ``COMMAND`` is a shellscript command, it is a good practice to put the actual code into a `function <#function>`__ and simply bind to the function name. This way it becomes significantly easier to test the function while editing, and the result is usually more readable as well. When ``COMMAND`` is a shellscript command, it is a good practice to put the actual code into a `function <#function>`__ and simply bind to the function name. This way it becomes significantly easier to test the function while editing, and the result is usually more readable as well.
@ -126,6 +128,8 @@ The following special input functions are available:
- ``execute``, run the current commandline - ``execute``, run the current commandline
- ``exit``, exit the shell
- ``forward-bigword``, move one whitespace-delimited word to the right - ``forward-bigword``, move one whitespace-delimited word to the right
- ``forward-char``, move one character to the right - ``forward-char``, move one character to the right

View file

@ -19,7 +19,7 @@ The fish_hg_prompt function displays information about the current Mercurial rep
`Mercurial <https://www.mercurial-scm.org/>`_ (``hg``) must be installed. `Mercurial <https://www.mercurial-scm.org/>`_ (``hg``) must be installed.
By default, only the current branch is shown because ``hg status`` can take be slow on large repository. You can enable a more informative prompt by setting the variable ``$fish_prompt_hg_show_informative_status``, for example:: By default, only the current branch is shown because ``hg status`` can be slow on a large repository. You can enable a more informative prompt by setting the variable ``$fish_prompt_hg_show_informative_status``, for example::
set --universal fish_prompt_hg_show_informative_status set --universal fish_prompt_hg_show_informative_status

View file

@ -0,0 +1,29 @@
.. _cmd-fish_status_to_signal:
fish_status_to_signal - Convert exit codes to human-friendly signals
====================================================================
Synopsis
--------
::
function fish_prompt
echo -n (fish_status_to_signal $pipestatus | string join '|') (prompt_pwd) '$ '
end
Description
-----------
``fish_status_to_signal`` converts exit codes to their corresponding human-friendly signals if one exists.
This is likely to be useful for prompts in conjunction with the ``$status`` and ``$pipestatus`` variables.
Example
-------
::
>_ sleep 5
^C⏎
>_ fish_status_to_signal $status
SIGINT

View file

@ -27,7 +27,7 @@ If ``--index`` or ``-n`` is given, each match is reported as a 1-based start pos
If ``--regex`` or ``-r`` is given, PATTERN is interpreted as a Perl-compatible regular expression, which does not have to match the entire STRING. For a regular expression containing capturing groups, multiple items will be reported for each match, one for the entire match and one for each capturing group. With this, only the matching part of the STRING will be reported, unless ``--entire`` is given. If ``--regex`` or ``-r`` is given, PATTERN is interpreted as a Perl-compatible regular expression, which does not have to match the entire STRING. For a regular expression containing capturing groups, multiple items will be reported for each match, one for the entire match and one for each capturing group. With this, only the matching part of the STRING will be reported, unless ``--entire`` is given.
When matching via regular expressions, ``string match`` automatically imports all named capturing groups (``(?<name>expression)``) as fish variables of the same name. It will create a variable in the default scope for each named capturing group, and set it to the value of the capturing group in the first matched argument. If a named capture group matched an empty string, the variable will be set to the empty string (like ``set var ""``). If it did not match, the variable will be set to nothing (like ``set var``). When ``--regex`` is used with ``--all``, this behavior changes. Each named variable will contain a list of matches, with the first match contained in the first element, the second match in the second, and so on. If the group was empty or did not match, the corresponding element will be an empty string. When matching via regular expressions, ``string match`` automatically sets variables for all named capturing groups (``(?<name>expression)``). It will create a variable with the name of the group, in the default scope, for each named capturing group, and set it to the value of the capturing group in the first matched argument. If a named capture group matched an empty string, the variable will be set to the empty string (like ``set var ""``). If it did not match, the variable will be set to nothing (like ``set var``). When ``--regex`` is used with ``--all``, this behavior changes. Each named variable will contain a list of matches, with the first match contained in the first element, the second match in the second, and so on. If the group was empty or did not match, the corresponding element will be an empty string.
If ``--invert`` or ``-v`` is used the selected lines will be only those which do not match the given glob pattern or regular expression. If ``--invert`` or ``-v`` is used the selected lines will be only those which do not match the given glob pattern or regular expression.

View file

@ -479,6 +479,7 @@ When fish is given a commandline, it expands the parameters before sending them
- :ref:`Brace expansion <expand-brace>`, to write lists with common pre- or suffixes in a shorter way - :ref:`Brace expansion <expand-brace>`, to write lists with common pre- or suffixes in a shorter way
- :ref:`Tilde expansion <expand-home>`, to turn the ``~`` at the beginning of paths into the path to the home directory - :ref:`Tilde expansion <expand-home>`, to turn the ``~`` at the beginning of paths into the path to the home directory
Parameter expansion is limited to 524288 items.
.. _expand-wildcard: .. _expand-wildcard:
@ -487,9 +488,9 @@ Wildcards ("Globbing")
When a parameter includes an :ref:`unquoted <quotes>` ``*`` star (or "asterisk") or a ``?`` question mark, fish uses it as a wildcard to match files. When a parameter includes an :ref:`unquoted <quotes>` ``*`` star (or "asterisk") or a ``?`` question mark, fish uses it as a wildcard to match files.
- ``*`` can match any string of characters not containing ``/``. This includes matching an empty string. - ``*`` matches any number of characters (including zero) in a file name, not including ``/``.
- ``**`` matches any string of characters. This includes matching an empty string. The matched string can include the ``/`` character; that is, it goes into subdirectories. If a wildcard string with ``**`` contains a ``/``, that ``/`` still needs to be matched. For example, ``**\/*.fish`` won't match ``.fish`` files directly in the PWD, only in subdirectories. In fish you should type ``**.fish`` to match files in the PWD as well as subdirectories. [#]_ - ``**`` matches any number of characters (including zero), and also descends into subdirectories. If ``**`` is a segment by itself, that segment may match zero times, for compatibility with other shells.
- ``?`` can match any single character except ``/``. This is deprecated and can be disabled via the ``qmark-noglob`` :ref:`feature flag<featureflags>`, so ``?`` will just be an ordinary character. - ``?`` can match any single character except ``/``. This is deprecated and can be disabled via the ``qmark-noglob`` :ref:`feature flag<featureflags>`, so ``?`` will just be an ordinary character.
@ -530,7 +531,6 @@ Examples::
end end
# Lists the .foo files, if any. # Lists the .foo files, if any.
.. [#] Unlike other shells, notably zsh.
.. [#] Technically, unix allows filenames with newlines, and this splits the ``find`` output on newlines. If you want to avoid that, use find's ``-print0`` option and :ref:`string split0<cmd-string-split0>`. .. [#] Technically, unix allows filenames with newlines, and this splits the ``find`` output on newlines. If you want to avoid that, use find's ``-print0`` option and :ref:`string split0<cmd-string-split0>`.
.. _expand-command-substitution: .. _expand-command-substitution:
@ -1993,7 +1993,11 @@ If a function named :ref:`fish_greeting <cmd-fish_greeting>` exists, it will be
Private mode Private mode
------------- -------------
fish supports launching in private mode via ``fish --private`` (or ``fish -P`` for short). In private mode, old history is not available and any interactive commands you execute will not be appended to the global history file, making it useful both for avoiding inadvertently leaking personal information (e.g. for screencasts) and when dealing with sensitive information to prevent it being persisted to disk. You can query the global variable ``fish_private_mode`` (``if set -q fish_private_mode ...``) if you would like to respect the user's wish for privacy and alter the behavior of your own fish scripts. If ``$fish_private_mode`` is set to a non-empty value, commands will not be written to the history file on disk.
You can also launch with ``fish --private`` (or ``fish -P`` for short). This both hides old history and prevents writing history to disk. This is useful to avoid leaking personal information (e.g. for screencasts) or when dealing with sensitive information.
You can query the variable ``fish_private_mode`` (``if set -q fish_private_mode ...``) if you would like to respect the user's wish for privacy and alter the behavior of your own fish scripts.
.. _event: .. _event:

View file

@ -59,7 +59,8 @@ div.sphinxsidebar {
float: none; float: none;
} }
/* On screens that are less than 700px wide remove the sidebar */ /* On screens that are less than 700px wide remove anything non-essential
- the sidebar, the gradient background, ... */
@media screen and (max-width: 700px) { @media screen and (max-width: 700px) {
div.sphinxsidebar { div.sphinxsidebar {
width: 100%; width: 100%;
@ -68,6 +69,13 @@ div.sphinxsidebar {
} }
div.content {margin-left: 0;} div.content {margin-left: 0;}
div.bodywrapper { margin: 0; } div.bodywrapper { margin: 0; }
div#fmain {
border-radius: 0px;
margin: 0;
-moz-box-shadow: 0;
-webkit-box-shadow: 0;
box-shadow: 0;
}
} }
div.sphinxsidebar h3, div.sphinxsidebar h4 { div.sphinxsidebar h3, div.sphinxsidebar h4 {

View file

@ -275,6 +275,8 @@ You can erase (or "delete") a variable with ``-e`` or ``--erase``
> env | grep MyVariable > env | grep MyVariable
(no output) (no output)
.. _tut-exports:
Exports (Shell Variables) Exports (Shell Variables)
------------------------- -------------------------
@ -290,6 +292,7 @@ To give a variable to an external command, it needs to be "exported". Unlike oth
It can also be unexported with ``--unexport`` or ``-u``. It can also be unexported with ``--unexport`` or ``-u``.
This works the other way around as well! If fish is started by something else, it inherits that parents exported variables. So if your terminal emulator starts fish, and it exports ``$LANG`` set to ``en_US.UTF-8``, fish will receive that setting. And whatever started your terminal emulator also gave *it* some variables that it will then pass on unless it specifically decides not to. This is how fish usually receives the values for things like ``$LANG``, ``$PATH`` and ``$TERM``, without you having to specify them again.
.. _tut-lists: .. _tut-lists:
@ -641,6 +644,8 @@ $PATH
``$PATH`` is an environment variable containing the directories that fish searches for commands. Unlike other shells, $PATH is a :ref:`list <tut-lists>`, not a colon-delimited string. ``$PATH`` is an environment variable containing the directories that fish searches for commands. Unlike other shells, $PATH is a :ref:`list <tut-lists>`, not a colon-delimited string.
Fish takes care to set ``$PATH`` to a default, but typically it is just inherited from fish's parent process and is set to a value that makes sense for the system - see :ref:`Exports <tut-exports>`.
To prepend /usr/local/bin and /usr/sbin to ``$PATH``, you can write:: To prepend /usr/local/bin and /usr/sbin to ``$PATH``, you can write::
> set PATH /usr/local/bin /usr/sbin $PATH > set PATH /usr/local/bin /usr/sbin $PATH

View file

@ -0,0 +1,38 @@
FROM ubuntu:18.04
ENV LANG C.UTF-8
ENV LC_ALL C.UTF-8
ENV CXXFLAGS="-fno-omit-frame-pointer -fsanitize=undefined -fsanitize=address" \
CC=clang-9 \
CXX=clang++-9 \
ASAN_OPTIONS=check_initialization_order=1:detect_stack_use_after_return=1:detect_leaks=1 \
UBSAN_OPTIONS=print_stacktrace=1:report_error_type=1
RUN apt-get update \
&& apt-get -y install \
build-essential \
cmake \
clang-9 \
gettext \
git \
libncurses5-dev \
locales \
ninja-build \
python3 \
python3-pexpect \
sudo \
&& locale-gen en_US.UTF-8
RUN groupadd -g 1000 fishuser \
&& useradd -p $(openssl passwd -1 fish) -d /home/fishuser -m -u 1000 -g 1000 fishuser \
&& adduser fishuser sudo \
&& mkdir -p /home/fishuser/fish-build \
&& mkdir /fish-source \
&& chown -R fishuser:fishuser /home/fishuser /fish-source
USER fishuser
WORKDIR /home/fishuser
COPY fish_run_tests.sh /
CMD /fish_run_tests.sh

1
share/completions/..fish Normal file
View file

@ -0,0 +1 @@
complete . -w source

View file

@ -0,0 +1,29 @@
function __fish_print_alternatives_names -d "Get the names of link groups in the alternatives system"
alternatives --list | cut -f 1 | string trim
end
# common options
complete -c alternatives -l verbose -d "Generate more comments about what alternatives is doing"
complete -c alternatives -l help -d "Give some usage information"
complete -c alternatives -l version -d "Tell which version of alternatives this is"
complete -c alternatives -l keep-missing -d "If new variant doesn't provide some files, keep previous links"
complete -c alternatives -l altdir -xa "(__fish_complete_directories)" -d "Specifies the alternatives directory"
complete -c alternatives -l admindir -xa "(__fish_complete_directories)" -d "Specifies the administrative directory"
# actions
complete -c alternatives -l install -r -d "Add a group of alternatives to the system"
complete -c alternatives -l slave -n "__fish_contains_opt install" -r -d "Add a slave link to the new group"
complete -c alternatives -l initscript -n "__fish_contains_opt install" -F -d "Add an initscript for the new group"
complete -c alternatives -l family -n "__fish_contains_opt install" -x -d "Set a family for the new group"
complete -c alternatives -l remove -ra "(__fish_print_alternatives_names)" -d "Remove an alternative and all of its associated slave links"
complete -c alternatives -l set -ra "(__fish_print_alternatives_names)" -d "Set link group to given path"
complete -c alternatives -l config -xa "(__fish_print_alternatives_names)" -d "Open menu to configure link group"
complete -c alternatives -l auto -xa "(__fish_print_alternatives_names)" -d "Switch the master symlink name to automatic mode"
complete -c alternatives -l display -xa "(__fish_print_alternatives_names)" -d "Display information about the link group"
complete -c alternatives -l list -f -d "Display information about all link groups"
complete -c alternatives -l remove-all -xa "(__fish_print_alternatives_names)" -d "Remove the whole link group name"
complete -c alternatives -l add-slave -ra "(__fish_print_alternatives_names)" -d "Add a slave link to an existing alternative"
complete -c alternatives -l remove-slave -ra "(__fish_print_alternatives_names)" -d "Remove slave from an existing alternative"

View file

@ -3,7 +3,7 @@ complete -c fzf -f
# Search mode # Search mode
complete -c fzf -l no-extended -d no-extended complete -c fzf -l no-extended -d no-extended
complete -c fzf -n 'string match "+*" -- (commandline -ct)' -a +x -d no-extended complete -c fzf -n 'string match "+*" -- (commandline -ct)' -a +x -d no-extended
complete -c fzf -s e -l --exact -d 'Enable exact-match' complete -c fzf -s e -l exact -d 'Enable exact-match'
complete -c fzf -n 'string match "+*" -- (commandline -ct)' -a +i -d 'case-sensitive match' complete -c fzf -n 'string match "+*" -- (commandline -ct)' -a +i -d 'case-sensitive match'
complete -c fzf -s i -d 'Case-insensitive match' complete -c fzf -s i -d 'Case-insensitive match'
complete -c fzf -l literal -d 'Do not normalize latin script letters for matching' complete -c fzf -l literal -d 'Do not normalize latin script letters for matching'

View file

@ -937,14 +937,14 @@ complete -f -c git -n '__fish_git_using_command show' -l oneline -d 'Shorthand f
complete -f -c git -n '__fish_git_using_command show' -l encoding -d 'Re-code the commit log message in the encoding' complete -f -c git -n '__fish_git_using_command show' -l encoding -d 'Re-code the commit log message in the encoding'
complete -f -c git -n '__fish_git_using_command show' -l expand-tabs -d 'Perform a tab expansion in the log message' complete -f -c git -n '__fish_git_using_command show' -l expand-tabs -d 'Perform a tab expansion in the log message'
complete -f -c git -n '__fish_git_using_command show' -l no-expand-tabs -d 'Do not perform a tab expansion in the log message' complete -f -c git -n '__fish_git_using_command show' -l no-expand-tabs -d 'Do not perform a tab expansion in the log message'
complete -f -c git -n '__fish_git_using_command show' -l notes -a '(__fish_git_refs)' -d 'Show the notes that annotate the commit' complete -f -c git -n '__fish_git_using_command show' -l notes -k -a '(__fish_git_refs)' -d 'Show the notes that annotate the commit'
complete -f -c git -n '__fish_git_using_command show' -l no-notes -d 'Do not show notes' complete -f -c git -n '__fish_git_using_command show' -l no-notes -d 'Do not show notes'
complete -f -c git -n '__fish_git_using_command show' -l show-signature -d 'Check the validity of a signed commit object' complete -f -c git -n '__fish_git_using_command show' -l show-signature -d 'Check the validity of a signed commit object'
### show-branch ### show-branch
complete -f -c git -n __fish_git_needs_command -a show-branch -d 'Shows the commits on branches' complete -f -c git -n __fish_git_needs_command -a show-branch -d 'Shows the commits on branches'
complete -f -c git -n '__fish_git_using_command show-branch' -a '(__fish_git_refs)' -d Rev complete -f -c git -n '__fish_git_using_command show-branch' -k -a '(__fish_git_refs)' -d Rev
# TODO options # TODO options
### add ### add
@ -1012,11 +1012,11 @@ complete -f -c git -n '__fish_git_using_command branch' -l no-merged -d 'List br
### cherry ### cherry
complete -f -c git -n __fish_git_needs_command -a cherry -d 'Find commits yet to be applied to upstream [upstream [head]]' complete -f -c git -n __fish_git_needs_command -a cherry -d 'Find commits yet to be applied to upstream [upstream [head]]'
complete -f -c git -n '__fish_git_using_command cherry' -s v -d 'Show the commit subjects next to the SHA1s' complete -f -c git -n '__fish_git_using_command cherry' -s v -d 'Show the commit subjects next to the SHA1s'
complete -f -c git -n '__fish_git_using_command cherry' -a '(__fish_git_refs)' -d Upstream complete -f -c git -n '__fish_git_using_command cherry' -k -a '(__fish_git_refs)' -d Upstream
### cherry-pick ### cherry-pick
complete -f -c git -n __fish_git_needs_command -a cherry-pick -d 'Apply the change introduced by an existing commit' complete -f -c git -n __fish_git_needs_command -a cherry-pick -d 'Apply the change introduced by an existing commit'
complete -f -c git -n '__fish_git_using_command cherry-pick' -a '(__fish_git_branches --no-merged)' complete -f -c git -n '__fish_git_using_command cherry-pick' -k -a '(__fish_git_ranges)'
# TODO: Filter further # TODO: Filter further
complete -f -c git -n '__fish_git_using_command cherry-pick; and __fish_git_possible_commithash' -ka '(__fish_git_commits)' complete -f -c git -n '__fish_git_using_command cherry-pick; and __fish_git_possible_commithash' -ka '(__fish_git_commits)'
complete -f -c git -n '__fish_git_using_command cherry-pick' -s e -l edit -d 'Edit the commit message prior to committing' complete -f -c git -n '__fish_git_using_command cherry-pick' -s e -l edit -d 'Edit the commit message prior to committing'
@ -1090,7 +1090,7 @@ complete -f -c git -n '__fish_git_using_command describe' -l first-parent -d 'Fo
### diff ### diff
complete -c git -n __fish_git_needs_command -a diff -d 'Show changes between commits, commit and working tree, etc' complete -c git -n __fish_git_needs_command -a diff -d 'Show changes between commits, commit and working tree, etc'
complete -c git -n '__fish_git_using_command diff; and not contains -- -- (commandline -opc)' -a '(__fish_git_ranges)' complete -c git -n '__fish_git_using_command diff; and not contains -- -- (commandline -opc)' -k -a '(__fish_git_ranges)'
complete -c git -n '__fish_git_using_command diff' -l cached -d 'Show diff of changes in the index' complete -c git -n '__fish_git_using_command diff' -l cached -d 'Show diff of changes in the index'
complete -c git -n '__fish_git_using_command diff' -l staged -d 'Show diff of changes in the index' complete -c git -n '__fish_git_using_command diff' -l staged -d 'Show diff of changes in the index'
complete -c git -n '__fish_git_using_command diff' -l no-index -d 'Compare two paths on the filesystem' complete -c git -n '__fish_git_using_command diff' -l no-index -d 'Compare two paths on the filesystem'
@ -1115,7 +1115,7 @@ end
### difftool ### difftool
complete -c git -n __fish_git_needs_command -a difftool -d 'Open diffs in a visual tool' complete -c git -n __fish_git_needs_command -a difftool -d 'Open diffs in a visual tool'
complete -c git -n '__fish_git_using_command difftool' -a '(__fish_git_ranges)' complete -c git -n '__fish_git_using_command difftool' -k -a '(__fish_git_ranges)'
complete -c git -n '__fish_git_using_command difftool' -l cached -d 'Visually show diff of changes in the index' complete -c git -n '__fish_git_using_command difftool' -l cached -d 'Visually show diff of changes in the index'
complete -f -c git -n '__fish_git_using_command difftool' -a '(__fish_git_files modified deleted)' complete -f -c git -n '__fish_git_using_command difftool' -a '(__fish_git_files modified deleted)'
complete -f -c git -n '__fish_git_using_command difftool' -s g -l gui -d 'Use `diff.guitool` instead of `diff.tool`' complete -f -c git -n '__fish_git_using_command difftool' -s g -l gui -d 'Use `diff.guitool` instead of `diff.tool`'
@ -1150,7 +1150,7 @@ complete -f -c git -n __fish_git_needs_command -a init -d 'Create an empty git r
### log ### log
complete -c git -n __fish_git_needs_command -a shortlog -d 'Show commit shortlog' complete -c git -n __fish_git_needs_command -a shortlog -d 'Show commit shortlog'
complete -c git -n __fish_git_needs_command -a log -d 'Show commit logs' complete -c git -n __fish_git_needs_command -a log -d 'Show commit logs'
complete -c git -n '__fish_git_using_command log; and not contains -- -- (commandline -opc)' -a '(__fish_git_ranges)' complete -c git -n '__fish_git_using_command log; and not contains -- -- (commandline -opc)' -k -a '(__fish_git_ranges)'
complete -c git -n '__fish_git_using_command log' -l follow -d 'Continue listing file history beyond renames' complete -c git -n '__fish_git_using_command log' -l follow -d 'Continue listing file history beyond renames'
complete -c git -n '__fish_git_using_command log' -l no-decorate -d 'Don\'t print ref names' complete -c git -n '__fish_git_using_command log' -l no-decorate -d 'Don\'t print ref names'
@ -1555,7 +1555,7 @@ complete -f -c git -n '__fish_git_using_command reset; and not contains -- -- (c
### restore and switch ### restore and switch
# restore options # restore options
complete -f -c git -n __fish_git_needs_command -a restore -d 'Restore working tree files' complete -f -c git -n __fish_git_needs_command -a restore -d 'Restore working tree files'
complete -f -c git -n '__fish_git_using_command restore' -r -s s -l source -d 'Specify the source tree used to restore the working tree' -a '(__fish_git_refs)' complete -f -c git -n '__fish_git_using_command restore' -r -s s -l source -d 'Specify the source tree used to restore the working tree' -k -a '(__fish_git_refs)'
complete -f -c git -n '__fish_git_using_command restore' -s p -l patch -d 'Interactive mode' complete -f -c git -n '__fish_git_using_command restore' -s p -l patch -d 'Interactive mode'
complete -f -c git -n '__fish_git_using_command restore' -s W -l worktree -d 'Restore working tree (default)' complete -f -c git -n '__fish_git_using_command restore' -s W -l worktree -d 'Restore working tree (default)'
complete -f -c git -n '__fish_git_using_command restore' -s S -l staged -d 'Restore the index' complete -f -c git -n '__fish_git_using_command restore' -s S -l staged -d 'Restore the index'

View file

@ -0,0 +1,45 @@
complete -c ldapsearch -s V -d 'Print version info'
complete -c ldapsearch -o VV -d 'Print version info and exit'
complete -c ldapsearch -s d -x -d 'Set the LDAP debug level'
complete -c ldapsearch -s n -d 'Show what would be done, but don\'t actually perform search'
complete -c ldapsearch -s v -d 'Run in verbose mode'
complete -c ldapsearch -s c -d 'Continuous operation mode'
complete -c ldapsearch -s u -d 'Include User Friendly Name of the Distinguished Name (DN)'
complete -c ldapsearch -s t -d 'Write retrieved non-printable values to temp files'
complete -o ldapsearch -o tt -d 'Write all retrieved values to temp files'
complete -c ldapsearch -s T -r -d 'Write temp files to specified directory'
complete -c ldapsearch -s F -x -d 'URL prefix for temp files'
complete -c ldapsearch -s A -d 'Retrieve attributes only'
complete -c ldapsearch -s L -d 'Search results are display in LDAP Data Interchange Format'
complete -c ldapsearch -s S -x -d 'Sort the entries returned based on attribute'
complete -c ldapsearch -s b -x -d 'Specify starting point for the search'
complete -c ldapsearch -s s -xa 'base one sub children' -d 'Specify the scope of the search'
complete -c ldapsearch -s a -xa 'never always search find' -d 'Specify how aliases dereferencing is done'
complete -c ldapsearch -s l -x -d 'Set max number of seconds for a search to complete'
complete -c ldapsearch -s z -x -d 'Set max number of entries for a search'
complete -c ldapsearch -s f -r -d 'Read lines from file, performing one search for each'
complete -c ldapsearch -s M -d 'Enable manage DSA IT control'
complete -c ldapsearch -o MM -d 'Enable manage DSA IT with critical control'
complete -c ldapsearch -s x -d 'Use simple authentication instead of SASL'
complete -c ldapsearch -s D -x -d 'Specify Distinguished Name to bind to the LDAP directory'
complete -c ldapsearch -s W -d 'Prompt for simple authentication'
complete -c ldapsearch -s w -x -d 'Set password for simple authentication'
complete -c ldapsearch -s y -r -d 'Use contents of file as password for simple authentication'
complete -c ldapsearch -s H -x -d 'Specify URI(s) referring to the ldap server(s)'
complete -c ldapsearch -s h -x -d 'Specify an alternate host'
complete -c ldapsearch -s p -x -d 'Specify an alternate TCP port'
complete -c ldapsearch -s P -xa '2 3' -d 'Specify the LDAP protocol version'
complete -c ldapsearch -s e -x -d 'Specify general extensions'
complete -c ldapsearch -s E -x -d 'Specify search extensions'
complete -c ldapsearch -s o -x -d 'Specify general options'
complete -c ldapsearch -s O -x -d 'Specify SASL security properties'
complete -c ldapsearch -s I -d 'Enable SASL Interactive mode'
complete -c ldapsearch -s Q -d 'Enable SASL Quiet mode'
complete -c ldapsearch -s N -d 'Do not use reverse DNS to canonicalize SASL host name'
complete -c ldapsearch -s U -x -d 'Specify the authentication ID for SASL bind'
complete -c ldapsearch -s R -x -d 'Specify the realm of authentication ID for SASL bind'
complete -c ldapsearch -s X -x -d 'Specify the requested authorization ID for SASL bind'
complete -c ldapsearch -s Y -x -d 'Specify the SASL mechanism to be used for authentication'
complete -c ldapsearch -s Z -d 'Issue StartTLS extended operation'
complete -c ldapsearch -o ZZ -d 'Issue StartTLS ectended operation only if succesful'

View file

@ -0,0 +1,48 @@
# losetup - Set up and control loop devices.
#
# This is part of the util-linux package.
# https://www.kernel.org/pub/linux/utils/util-linux
function __fish_print_losetup_list_output
printf "%s\t%s\n" \
NAME "Loop device name" \
AUTOCLEAR "Autoclear flag set" \
BACK-FILE "Device backing file" \
BACK-INO "Backing file inode number" \
BACK-MAJ:MIN "Backing file major:minor device number" \
MAJ:MIN "Loop device major:minor number" \
OFFSET "Offset from the beginning" \
PARTSCAN "Partscan flag set" \
RO "Read-only device" \
SIZELIMIT "Size limit of the file in bytes" \
DIO "Access backing file with direct-io" \
LOG-SEC "Logical sector size in bytes"
end
function __fish_print_losetup_attached
losetup --list --raw --noheadings --output NAME,BACK-FILE | string replace ' ' \t
end
complete -c losetup -s a -l all -d "List all used devices"
complete -c losetup -s d -l detach -x -a "(__fish_print_losetup_attached)" -d "Detach one or more devices"
complete -c losetup -s D -l detach-all -d "Detach all used devices"
complete -c losetup -s f -l find -d "Find first unused device"
complete -c losetup -s c -l set-capacity -x -a "(__fish_print_losetup_attached)" -d "Resize the device"
complete -c losetup -s j -l associated -r -d "List all devices associated with given file"
complete -c losetup -s L -l nooverlap -d "Avoid possible conflict between devices"
complete -c losetup -s o -l offset -x -d "Start at given offset into file"
complete -c losetup -l sizelimit -x -d "Device is limited to give bytes of the file"
complete -c losetup -s b -l sector-size -x -d "Set the logical sector size"
complete -c losetup -s P -l partscan -d "Create a partitioned loop device"
complete -c losetup -s r -l read-only -d "Set up a read-only loop device"
complete -c losetup -l direct-io -x -a "on off" -d "open backing file with O_DIRECT"
complete -c losetup -l show -d "Print device name after setup"
complete -c losetup -s v -l verbose -d "Verbose mode"
complete -c losetup -s J -l json -d "Use JSON --list output format"
complete -c losetup -s l -l list -d "List info about all or specified"
complete -c losetup -s n -l noheadings -d "Don't print headings for --list output"
complete -c losetup -s O -l output -x -a "(__fish_complete_list , __fish_print_losetup_list_output)" -d "Specify columns to output for --list"
complete -c losetup -l output-all -d "Output all columns"
complete -c losetup -l raw -d "Use raw --list output format"
complete -c losetup -s h -l help -d "Display help"
complete -c losetup -s V -l version -d "Display version"

View file

@ -1,6 +1,14 @@
# Completions for pkgng package manager # Completions for pkgng package manager
if uname | not string match -q FreeBSD # Solaris has a thing called "pkg"; it works quite differently and spews errors when called here.
# There are multiple SunOS-derived distributions and not all of them have `SunOS` in their name (and
# some of them also use pkgsrc and have a `pkg`).
#
# Additionally, this particular script is intended to complete the pkgng "Next Generation" package
# manager initially developed for FreeBSD though now available on a few other BSDs. From here on
# out, maintainers can assume we are specifically talking about the (Free)BSD `pkg` command being
# executed on a BSD system, rather than just work with "not SunOS".
if ! uname | string match -irq bsd
exit exit
end end
@ -49,10 +57,12 @@ complete -c pkg -n __fish_pkg_subcommand -s 4 -d "Use IPv4"
complete -c pkg -n __fish_pkg_subcommand -s 6 -d "Use IPv6" complete -c pkg -n __fish_pkg_subcommand -s 6 -d "Use IPv6"
complete -c pkg -n __fish_pkg_subcommand -xa add -d "Install package file" complete -c pkg -n __fish_pkg_subcommand -xa add -d "Install package file"
complete -c pkg -n __fish_pkg_subcommand -xa alias -d "List the command line aliases"
complete -c pkg -n __fish_pkg_subcommand -xa annotate -d "Modify annotations on packages" complete -c pkg -n __fish_pkg_subcommand -xa annotate -d "Modify annotations on packages"
complete -c pkg -n __fish_pkg_subcommand -xa audit -d "Audit installed packages" complete -c pkg -n __fish_pkg_subcommand -xa audit -d "Audit installed packages"
complete -c pkg -n __fish_pkg_subcommand -xa autoremove -d "Delete unneeded packages" complete -c pkg -n __fish_pkg_subcommand -xa autoremove -d "Delete unneeded packages"
complete -c pkg -n __fish_pkg_subcommand -xa backup -d "Dump package database" complete -c pkg -n __fish_pkg_subcommand -xa backup -d "Dump package database"
complete -c pkg -n __fish_pkg_subcommand -xa bootstrap -d "Install pkg(8) from remote repository"
complete -c pkg -n __fish_pkg_subcommand -xa check -d "Check installed packages" complete -c pkg -n __fish_pkg_subcommand -xa check -d "Check installed packages"
complete -c pkg -n __fish_pkg_subcommand -xa clean -d "Clean local cache" complete -c pkg -n __fish_pkg_subcommand -xa clean -d "Clean local cache"
complete -c pkg -n __fish_pkg_subcommand -xa convert -d "Convert package from pkg_add format" complete -c pkg -n __fish_pkg_subcommand -xa convert -d "Convert package from pkg_add format"
@ -85,15 +95,42 @@ complete -c pkg -n __fish_pkg_subcommand -xa which -d "Check which package provi
# add # add
complete -c pkg -n '__fish_pkg_is add install' -s A -l automatic -d "Mark packages as automatic" complete -c pkg -n '__fish_pkg_is add install' -s A -l automatic -d "Mark packages as automatic"
complete -c pkg -n '__fish_pkg_is add install' -s f -l force -d "Force installation even when installed" complete -c pkg -n '__fish_pkg_is add bootstrap install' -s f -l force -d "Force installation even when installed"
complete -c pkg -n '__fish_pkg_is add' -s I -l no-scripts -d "Disable installation scripts" complete -c pkg -n '__fish_pkg_is add' -s I -l no-scripts -d "Disable installation scripts"
complete -c pkg -n '__fish_pkg_is add' -s M -l accept-missing -d "Force installation with missing dependencies" complete -c pkg -n '__fish_pkg_is add' -s M -l accept-missing -d "Force installation with missing dependencies"
complete -c pkg -n '__fish_pkg_is add autoremove clean delete remove install update' -s q -l quiet -d "Force quiet output" complete -c pkg -n '__fish_pkg_is add alias autoremove clean delete remove install update' -s q -l quiet -d "Force quiet output"
# alias
complete -c pkg -n '__fish_pkg_is alias' -xa '(pkg alias -lq)'
complete -c pkg -n '__fish_pkg_is alias' -s l -l list -d "Print all aliases without their pkg(8) arguments"
# autoremove # autoremove
complete -c pkg -n '__fish_pkg_is autoremove clean delete remove install upgrade' -s n -l dry-run -d "Do not make changes" complete -c pkg -n '__fish_pkg_is autoremove clean delete remove install upgrade' -s n -l dry-run -d "Do not make changes"
complete -c pkg -n '__fish_pkg_is autoremove clean delete remove install' -s y -l yes -d "Assume yes when asked for confirmation" complete -c pkg -n '__fish_pkg_is autoremove clean delete remove install' -s y -l yes -d "Assume yes when asked for confirmation"
# bootstrap
complete -c pkg -n '__fish_pkg_is bootstrap' -f
# check
set -l has_check_opt '__fish_contains_opt -s B shlibs -s d dependencies -s s checksums -s r recompute'
set -l has_all_opt '__fish_contains_opt -s a all'
complete -c pkg -n "__fish_pkg_is check" -f
complete -c pkg -n "__fish_pkg_is check; and not $has_check_opt" -xa "-B -d -s -r"
complete -c pkg -n "__fish_pkg_is check; and not $has_check_opt" -s B -l shlibs -d "Regenerate library dependency metadata"
complete -c pkg -n "__fish_pkg_is check; and not $has_check_opt" -s d -l dependencies -d "Check for and install missing dependencies"
complete -c pkg -n "__fish_pkg_is check; and not $has_check_opt" -s r -l recompute -d "Recalculate and set the checksums of installed packages"
complete -c pkg -n "__fish_pkg_is check; and not $has_check_opt" -s s -l checksums -d "Detect installed packages with invalid checksums"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt" -s n -l dry-run -d "Do not make changes"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt" -s q -l quiet -d "Force quiet output"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt" -s v -l verbose -d "Provide verbose output"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt" -s y -l yes -d "Assume yes when asked for confirmation"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt; and not $has_all_opt" -xa '(pkg query "%n")'
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt; and not $has_all_opt" -s a -l all -d "Process all packages"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt; and not $has_all_opt" -s C -l case-sensitive -d "Case sensitive packages"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt; and not $has_all_opt" -s g -l glob -d "Treat the package name as shell glob"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt; and not $has_all_opt" -s i -l case-insensitive -d "Case insensitive packages"
complete -c pkg -n "__fish_pkg_is check; and $has_check_opt; and not $has_all_opt" -s x -l regex -d "Treat the package name as regular expression"
# clean # clean
complete -c pkg -n '__fish_pkg_is clean' -s a -l all -d "Delete all cached packages" complete -c pkg -n '__fish_pkg_is clean' -s a -l all -d "Delete all cached packages"

View file

@ -1,11 +1,11 @@
complete -c rustc -s h -l help complete -c rustc -s h -l help
complete -c rustc -x -l cfg complete -c rustc -x -l cfg
complete -c rustc -r -s L -a 'dylib= static= framework=' complete -c rustc -r -s L -a 'dependency= crate= native= framework= all='
complete -c rustc -x -s l -a 'dylib= static= framework=' complete -c rustc -x -s l -a 'dylib= static= framework='
complete -c rustc -x -l crate-type -a 'bin lib rlib dylib staticlib' complete -c rustc -x -l crate-type -a 'bin lib rlib dylib staticlib proc-macro'
complete -c rustc -r -l crate-name complete -c rustc -r -l crate-name
complete -c rustc -x -l emit -a 'asm llvm-bc llvm-ir obj link dep-info' complete -c rustc -x -l emit -a 'asm llvm-bc llvm-ir obj link dep-info metadata mir'
complete -c rustc -x -l print -a 'crate-name file-names sysroot' complete -c rustc -x -l print -a 'crate-name file-names sysroot'
complete -c rustc -s g complete -c rustc -s g
complete -c rustc -s O complete -c rustc -s O

View file

@ -0,0 +1,2 @@
complete source -k -xa '(__fish_complete_suffix .fish)'
complete source -s h -l help -d 'Display help and exit'

View file

@ -5,7 +5,7 @@ complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a status
complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a show -d 'Show properties of systemd-timedated' complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a show -d 'Show properties of systemd-timedated'
complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a set-time -d 'Set system time' complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a set-time -d 'Set system time'
complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a set-timezone -d 'Set system time zone' complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a set-timezone -d 'Set system time zone'
complete -c timedatectl -n "__fish_seen_subcommand_from set-timezone" -a (timedatectl list-timezones) complete -c timedatectl -n "__fish_seen_subcommand_from set-timezone" -a "(timedatectl list-timezones)"
complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a list-timezones -d 'Show known time zones' complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a list-timezones -d 'Show known time zones'
complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a set-local-rtc -d 'Control whether RTC is in local time' complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" -a set-local-rtc -d 'Control whether RTC is in local time'
complete -c timedatectl -n "__fish_seen_subcommand_from set-local-rtc" -a 'true false' complete -c timedatectl -n "__fish_seen_subcommand_from set-local-rtc" -a 'true false'

View file

@ -94,7 +94,7 @@ function __fish_config_interactive -d "Initializations that should be performed
# Run python directly in the background and swallow all output # Run python directly in the background and swallow all output
$python $update_args >/dev/null 2>&1 & $python $update_args >/dev/null 2>&1 &
# Then disown the job so that it continues to run in case of an early exit (#6269) # Then disown the job so that it continues to run in case of an early exit (#6269)
disown >/dev/null 2>&1 disown $last_pid >/dev/null 2>&1
end end
end end
end end
@ -247,6 +247,10 @@ function __fish_config_interactive -d "Initializations that should be performed
if set -q VTE_VERSION if set -q VTE_VERSION
return return
end end
# Same for alacritty
if string match -q -- 'alacritty*' $TERM
return
end
commandline -f repaint >/dev/null 2>/dev/null commandline -f repaint >/dev/null 2>/dev/null
end end

View file

@ -23,7 +23,7 @@ function __fish_print_pipestatus --description "Print pipestatus for prompt"
# SIGPIPE (141 = 128 + 13) is usually not a failure, see #6375. # SIGPIPE (141 = 128 + 13) is usually not a failure, see #6375.
if not contains $last_status 0 141 if not contains $last_status 0 141
set -l sep $brace_sep_color$separator$status_color set -l sep $brace_sep_color$separator$status_color
set -l last_pipestatus_string (__fish_status_to_signal $argv | string join "$sep") set -l last_pipestatus_string (fish_status_to_signal $argv | string join "$sep")
set -l last_status_string "" set -l last_status_string ""
if test $last_status -ne $argv[-1] if test $last_status -ne $argv[-1]
set last_status_string " "$status_color$last_status set last_status_string " "$status_color$last_status

View file

@ -38,7 +38,10 @@ function fish_clipboard_paste
# so we don't trigger ignoring history. # so we don't trigger ignoring history.
set data[1] (string trim -l -- $data[1]) set data[1] (string trim -l -- $data[1])
end end
if test -n "$data" if test -n "$data"
commandline -f begin-undo-group
commandline -i -- $data commandline -i -- $data
commandline -f end-undo-group
end end
end end

View file

@ -1,4 +1,4 @@
function __fish_status_to_signal --description "Print signal name from argument (\$status), or just argument" function fish_status_to_signal --description "Print signal name from argument (\$status), or just argument"
for arg in $argv for arg in $argv
if test $arg -gt 128 if test $arg -gt 128
set -l signals SIGHUP SIGINT SIGQUIT SIGILL SIGTRAP SIGABRT SIGBUS \ set -l signals SIGHUP SIGINT SIGQUIT SIGILL SIGTRAP SIGABRT SIGBUS \

View file

@ -199,7 +199,7 @@ function help --description 'Show help for the fish shell'
printf (_ 'help: Help is being displayed in %s.\n') $fish_browser[1] printf (_ 'help: Help is being displayed in %s.\n') $fish_browser[1]
end end
$fish_browser $page_url & $fish_browser $page_url &
disown disown $last_pid >/dev/null 2>&1
else else
# Work around lynx bug where <div class="contents"> always has the same formatting as links (unreadable) # Work around lynx bug where <div class="contents"> always has the same formatting as links (unreadable)
# by using a custom style sheet. See https://github.com/fish-shell/fish-shell/issues/4170 # by using a custom style sheet. See https://github.com/fish-shell/fish-shell/issues/4170

View file

@ -1,7 +1,6 @@
body { body {
background: linear-gradient(to bottom, #a7cfdf 0%, #23538a 100%); background: linear-gradient(to bottom, #a7cfdf 0%, #23538a 100%);
font-family: "Source Code Pro", "DejaVu Sans Mono", Menlo, "Ubuntu Mono", Consolas, Monaco, font-family: monospace, fixed;
"Lucida Console", monospace, fixed;
color: #222; color: #222;
min-height: 100vh; /* at least 1 screen high - to prevent the gradient from running out on a short tab */ min-height: 100vh; /* at least 1 screen high - to prevent the gradient from running out on a short tab */
width: 90%; width: 90%;

View file

@ -256,6 +256,108 @@ var color_scheme_fish_default = {
}; };
var ayuTheme = {
ayu_dark: {
'accent': 'E6B450',
'bg': '0A0E14',
'fg': 'B3B1AD',
'ui': '4D5566',
'tag': '39BAE6',
'func': 'FFB454',
'entity': '59C2FF',
'string': 'C2D94C',
'regexp': '95E6CB',
'markup': 'F07178',
'keyword': 'FF8F40',
'special': 'E6B673',
'comment': '626A73',
'constant': 'FFEE99',
'operator': 'F29668',
'error': 'FF3333',
},
ayu_light: {
'accent': 'FF9940',
'bg': 'FAFAFA',
'fg': '575F66',
'ui': '8A9199',
'tag': '55B4D4',
'func': 'F2AE49',
'entity': '399EE6',
'string': '86B300',
'regexp': '4CBF99',
'markup': 'F07171',
'keyword': 'FA8D3E',
'special': 'E6BA7E',
'comment': 'ABB0B6',
'constant': 'A37ACC',
'operator': 'ED9366',
'error': 'F51818',
},
ayu_mirage: {
'accent': 'FFCC66',
'bg': '1F2430',
'fg': 'CBCCC6',
'ui': '707A8C',
'tag': '5CCFE6',
'func': 'FFD580',
'entity': '73D0FF',
'string': 'BAE67E',
'regexp': '95E6CB',
'markup': 'F28779',
'keyword': 'FFA759',
'special': 'FFE6B3',
'comment': '5C6773',
'constant': 'D4BFFF',
'operator': 'F29E74',
'error': 'FF3333',
},
apply: function(theme, receiver) {
receiver['preferred_background'] = theme.bg
receiver['autosuggestion'] = theme.ui
receiver['command'] = theme.tag
receiver['comment'] = theme.comment
receiver['cwd'] = theme.entity
receiver['end'] = theme.operator
receiver['error'] = theme.error
receiver['escape'] = theme.regexp
receiver['match'] = theme.markup
receiver['normal'] = theme.fg
receiver['operator'] = theme.accent
receiver['param'] = theme.fg
receiver['quote'] = theme.string
receiver['redirection'] = theme.constant
receiver['search_match'] = theme.accent
receiver['selection'] = theme.accent
receiver['colors'] = []
for (var key in theme) receiver['colors'].push(theme[key])
},
}
// ayu Light
var color_scheme_ayu_light = {
name: 'ayu Light',
url: 'https://github.com/dempfi/ayu',
}
ayuTheme.apply(ayuTheme.ayu_light, color_scheme_ayu_light)
// ayu Dark
var color_scheme_ayu_dark = {
name: 'ayu Dark',
url: 'https://github.com/dempfi/ayu',
}
ayuTheme.apply(ayuTheme.ayu_dark, color_scheme_ayu_dark)
// ayu Mirage
var color_scheme_ayu_mirage = {
name: 'ayu Mirage',
url: 'https://github.com/dempfi/ayu',
}
ayuTheme.apply(ayuTheme.ayu_mirage, color_scheme_ayu_mirage)
var TomorrowTheme = { var TomorrowTheme = {
tomorrow_night: {'Background': '1d1f21', 'Current Line': '282a2e', 'Selection': '373b41', 'Foreground': 'c5c8c6', 'Comment': '969896', 'Red': 'cc6666', 'Orange': 'de935f', 'Yellow': 'f0c674', 'Green': 'b5bd68', 'Aqua': '8abeb7', 'Blue': '81a2be', 'Purple': 'b294bb' tomorrow_night: {'Background': '1d1f21', 'Current Line': '282a2e', 'Selection': '373b41', 'Foreground': 'c5c8c6', 'Comment': '969896', 'Red': 'cc6666', 'Orange': 'de935f', 'Yellow': 'f0c674', 'Green': 'b5bd68', 'Aqua': '8abeb7', 'Blue': '81a2be', 'Purple': 'b294bb'
}, },

View file

@ -79,7 +79,21 @@ controllers.controller("colorsController", function($scope, $http) {
$scope.sampleTerminalBackgroundColors = ['white', '#' + solarized.base3, '#300', '#003', '#' + solarized.base03, '#232323', '#'+nord.nord0, 'black']; $scope.sampleTerminalBackgroundColors = ['white', '#' + solarized.base3, '#300', '#003', '#' + solarized.base03, '#232323', '#'+nord.nord0, 'black'];
/* Array of FishColorSchemes */ /* Array of FishColorSchemes */
$scope.colorSchemes = [color_scheme_fish_default, color_scheme_solarized_light, color_scheme_solarized_dark, color_scheme_tomorrow, color_scheme_tomorrow_night, color_scheme_tomorrow_night_bright, color_scheme_nord, color_scheme_base16_default_dark, color_scheme_base16_default_light, color_scheme_base16_eighties]; $scope.colorSchemes = [
color_scheme_fish_default,
color_scheme_ayu_light,
color_scheme_ayu_dark,
color_scheme_ayu_mirage,
color_scheme_solarized_light,
color_scheme_solarized_dark,
color_scheme_tomorrow,
color_scheme_tomorrow_night,
color_scheme_tomorrow_night_bright,
color_scheme_nord,
color_scheme_base16_default_dark,
color_scheme_base16_default_light,
color_scheme_base16_eighties
];
for (var i=0; i < additional_color_schemes.length; i++) for (var i=0; i < additional_color_schemes.length; i++)
$scope.colorSchemes.push(additional_color_schemes[i]) $scope.colorSchemes.push(additional_color_schemes[i])
@ -301,7 +315,7 @@ controllers.controller("historyController", function($scope, $http, $timeout) {
$scope.prevPage = function () { $scope.prevPage = function () {
$scope.currentPage = Math.max($scope.currentPage - 1, 0); $scope.currentPage = Math.max($scope.currentPage - 1, 0);
}; };
$scope.nextPage = function () { $scope.nextPage = function () {
$scope.currentPage = Math.min($scope.currentPage + 1, $scope.currentPage = Math.min($scope.currentPage + 1,
$scope.filteredItemPages.length - 1); $scope.filteredItemPages.length - 1);

View file

@ -1,7 +1,6 @@
from __future__ import unicode_literals from __future__ import unicode_literals
from __future__ import print_function from __future__ import print_function
import binascii import binascii
import cgi
try: try:
from html import escape as escape_html from html import escape as escape_html
@ -1315,18 +1314,30 @@ class FishConfigHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
return self.send_error(403) return self.send_error(403)
self.path = p self.path = p
ctype, pdict = cgi.parse_header(self.headers["content-type"]) # This is cheesy, we want just the actual content-type.
# In some cases it'll give us the encoding as well,
# ("application/json;charset=utf-8")
# but we don't currently care.
ctype = self.headers["content-type"].split(";")[0]
if ctype == "multipart/form-data": if ctype == "application/x-www-form-urlencoded":
postvars = cgi.parse_multipart(self.rfile, pdict)
elif ctype == "application/x-www-form-urlencoded":
length = int(self.headers["content-length"]) length = int(self.headers["content-length"])
url_str = self.rfile.read(length).decode("utf-8") url_str = self.rfile.read(length).decode("utf-8")
postvars = parse_qs(url_str, keep_blank_values=1) postvars = parse_qs(url_str, keep_blank_values=1)
elif ctype == "application/json": elif ctype == "application/json":
length = int(self.headers["content-length"]) length = int(self.headers["content-length"])
url_str = self.rfile.read(length).decode(pdict["charset"]) # This used to use the provided encoding, but we use utf-8
# all around the place and nobody has ever complained.
#
# If any other encoding is received this will raise a UnicodeError,
# which will throw us out of the function and should at most exit webconfig.
# If that happens to anyone we expect bug reports.
url_str = self.rfile.read(length).decode("utf-8")
postvars = json.loads(url_str) postvars = json.loads(url_str)
elif ctype == "multipart/form-data":
# This used to be a thing, as far as I could find there's
# no use anymore, but let's keep an error around just in case.
return self.send_error(500)
else: else:
postvars = {} postvars = {}

View file

@ -32,7 +32,7 @@ static const wcstring var_name_prefix = L"_flag_";
#define BUILTIN_ERR_INVALID_OPT_SPEC _(L"%ls: Invalid option spec '%ls' at char '%lc'\n") #define BUILTIN_ERR_INVALID_OPT_SPEC _(L"%ls: Invalid option spec '%ls' at char '%lc'\n")
struct option_spec_t { struct option_spec_t {
const wchar_t short_flag; wchar_t short_flag;
wcstring long_flag; wcstring long_flag;
wcstring validation_command; wcstring validation_command;
wcstring_list_t vals; wcstring_list_t vals;
@ -208,14 +208,14 @@ static bool parse_flag_modifiers(const argparse_cmd_opts_t &opts, const option_s
/// Parse the text following the short flag letter. /// Parse the text following the short flag letter.
static bool parse_option_spec_sep(argparse_cmd_opts_t &opts, const option_spec_ref_t &opt_spec, static bool parse_option_spec_sep(argparse_cmd_opts_t &opts, const option_spec_ref_t &opt_spec,
const wcstring &option_spec, const wchar_t **opt_spec_str, const wcstring &option_spec, const wchar_t **opt_spec_str,
io_streams_t &streams) { wchar_t &counter, io_streams_t &streams) {
const wchar_t *s = *opt_spec_str; const wchar_t *s = *opt_spec_str;
if (*(s - 1) == L'#') { if (*(s - 1) == L'#') {
if (*s != L'-') { if (*s != L'-') {
streams.err.append_format( // Long-only!
_(L"%ls: Short flag '#' must be followed by '-' and a long name\n"), s--;
opts.name.c_str()); opt_spec->short_flag = counter;
return false; counter++;
} }
if (opts.implicit_int_flag) { if (opts.implicit_int_flag) {
streams.err.append_format(_(L"%ls: Implicit int flag '%lc' already defined\n"), streams.err.append_format(_(L"%ls: Implicit int flag '%lc' already defined\n"),
@ -250,9 +250,17 @@ static bool parse_option_spec_sep(argparse_cmd_opts_t &opts, const option_spec_r
opt_spec->num_allowed = 1; // mandatory arg and can appear only once opt_spec->num_allowed = 1; // mandatory arg and can appear only once
s++; // the struct is initialized assuming short_flag_valid should be true s++; // the struct is initialized assuming short_flag_valid should be true
} else { } else {
// Long flag name not allowed if second char isn't '/', '-' or '#' so just check for if (*s != L'!' && *s != L'?' && *s != L'=') {
// behavior modifier chars. // No short flag separator and no other modifiers, so this is a long only option.
if (!parse_flag_modifiers(opts, opt_spec, option_spec, &s, streams)) return false; // Since getopt needs a wchar, we have a counter that we count up.
opt_spec->short_flag_valid = false;
s--;
opt_spec->short_flag = counter;
counter++;
} else {
// Try to parse any other flag modifiers
if (!parse_flag_modifiers(opts, opt_spec, option_spec, &s, streams)) return false;
}
} }
*opt_spec_str = s; *opt_spec_str = s;
@ -261,10 +269,12 @@ static bool parse_option_spec_sep(argparse_cmd_opts_t &opts, const option_spec_r
/// This parses an option spec string into a struct option_spec. /// This parses an option spec string into a struct option_spec.
static bool parse_option_spec(argparse_cmd_opts_t &opts, //!OCLINT(high npath complexity) static bool parse_option_spec(argparse_cmd_opts_t &opts, //!OCLINT(high npath complexity)
const wcstring &option_spec, io_streams_t &streams) { const wcstring &option_spec, wchar_t &counter,
io_streams_t &streams) {
if (option_spec.empty()) { if (option_spec.empty()) {
streams.err.append_format(_(L"%ls: An option spec must have a short flag letter\n"), streams.err.append_format(
opts.name.c_str()); _(L"%ls: An option spec must have at least a short or a long flag\n"),
opts.name.c_str());
return false; return false;
} }
@ -278,7 +288,7 @@ static bool parse_option_spec(argparse_cmd_opts_t &opts, //!OCLINT(high npath c
std::unique_ptr<option_spec_t> opt_spec(new option_spec_t{*s++}); std::unique_ptr<option_spec_t> opt_spec(new option_spec_t{*s++});
// Try parsing stuff after the short flag. // Try parsing stuff after the short flag.
if (*s && !parse_option_spec_sep(opts, opt_spec, option_spec, &s, streams)) { if (*s && !parse_option_spec_sep(opts, opt_spec, option_spec, &s, counter, streams)) {
return false; return false;
} }
@ -315,22 +325,32 @@ static int collect_option_specs(argparse_cmd_opts_t &opts, int *optind, int argc
io_streams_t &streams) { io_streams_t &streams) {
wchar_t *cmd = argv[0]; wchar_t *cmd = argv[0];
// A counter to give short chars to long-only options because getopt needs that.
// Luckily we have wgetopt so we can use wchars - this is one of the private use areas so we
// have 6400 options available.
wchar_t counter = static_cast<wchar_t>(0xE000);
while (true) { while (true) {
if (std::wcscmp(L"--", argv[*optind]) == 0) { if (std::wcscmp(L"--", argv[*optind]) == 0) {
++*optind; ++*optind;
break; break;
} }
if (!parse_option_spec(opts, argv[*optind], streams)) { if (!parse_option_spec(opts, argv[*optind], counter, streams)) {
return STATUS_CMD_ERROR; return STATUS_CMD_ERROR;
} }
if (++*optind == argc) { if (++*optind == argc) {
streams.err.append_format(_(L"%ls: Missing -- separator\n"), cmd); streams.err.append_format(_(L"%ls: Missing -- separator\n"), cmd);
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;
} }
} }
// Check for counter overreach once at the end because this is very unlikely to ever be reached.
if (counter > static_cast<wchar_t>(0xF8FF)) {
streams.err.append_format(_(L"%ls: Too many long-only options\n"), cmd);
return STATUS_INVALID_ARGS;
}
if (opts.options.empty()) { if (opts.options.empty()) {
streams.err.append_format(_(L"%ls: No option specs were provided\n"), cmd); streams.err.append_format(_(L"%ls: No option specs were provided\n"), cmd);
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;

View file

@ -36,6 +36,9 @@ enum {
APPEND_MODE // insert at end of current token/command/buffer APPEND_MODE // insert at end of current token/command/buffer
}; };
/// Handle a single readline_cmd_t command out-of-band.
void reader_handle_command(readline_cmd_t cmd);
/// Replace/append/insert the selection with/at/after the specified string. /// Replace/append/insert the selection with/at/after the specified string.
/// ///
/// \param begin beginning of selection /// \param begin beginning of selection
@ -302,8 +305,18 @@ maybe_t<int> builtin_commandline(parser_t &parser, io_streams_t &streams, wchar_
if (mc == rl::repaint_mode || mc == rl::force_repaint || mc == rl::repaint) { if (mc == rl::repaint_mode || mc == rl::force_repaint || mc == rl::repaint) {
if (ld.is_repaint) continue; if (ld.is_repaint) continue;
} }
// Inserts the readline function at the back of the queue.
reader_queue_ch(*mc); // HACK: Execute these right here and now so they can affect any insertions/changes
// made via bindings. The correct solution is to change all `commandline`
// insert/replace operations into readline functions with associated data, so that
// all queued `commandline` operations - including buffer modifications - are
// executed in order
if (mc == rl::begin_undo_group || mc == rl::end_undo_group) {
reader_handle_command(*mc);
} else {
// Inserts the readline function at the back of the queue.
reader_queue_ch(*mc);
}
} else { } else {
streams.err.append_format(_(L"%ls: Unknown input function '%ls'"), cmd, argv[i]); streams.err.append_format(_(L"%ls: Unknown input function '%ls'"), cmd, argv[i]);
builtin_print_error_trailer(parser, streams.err, cmd); builtin_print_error_trailer(parser, streams.err, cmd);

View file

@ -205,6 +205,7 @@ maybe_t<int> builtin_complete(parser_t &parser, io_streams_t &streams, wchar_t *
} }
case 'd': { case 'd': {
desc = w.woptarg; desc = w.woptarg;
assert(desc);
break; break;
} }
case 'u': { case 'u': {
@ -249,6 +250,7 @@ maybe_t<int> builtin_complete(parser_t &parser, io_streams_t &streams, wchar_t *
} }
case 'a': { case 'a': {
comp = w.woptarg; comp = w.woptarg;
assert(comp);
break; break;
} }
case 'e': { case 'e': {
@ -257,6 +259,7 @@ maybe_t<int> builtin_complete(parser_t &parser, io_streams_t &streams, wchar_t *
} }
case 'n': { case 'n': {
condition = w.woptarg; condition = w.woptarg;
assert(condition);
break; break;
} }
case 'w': { case 'w': {

View file

@ -28,6 +28,8 @@ static int parse_cmd_opts(echo_cmd_opts_t &opts, int *optind, int argc, wchar_t
wchar_t *cmd = argv[0]; wchar_t *cmd = argv[0];
int opt; int opt;
wgetopter_t w; wgetopter_t w;
echo_cmd_opts_t oldopts = opts;
int oldoptind = 0;
while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, nullptr)) != -1) { while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, nullptr)) != -1) {
switch (opt) { switch (opt) {
case 'n': { case 'n': {
@ -51,6 +53,7 @@ static int parse_cmd_opts(echo_cmd_opts_t &opts, int *optind, int argc, wchar_t
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;
} }
case '?': { case '?': {
opts = oldopts;
*optind = w.woptind - 1; *optind = w.woptind - 1;
return STATUS_CMD_OK; return STATUS_CMD_OK;
} }
@ -58,6 +61,17 @@ static int parse_cmd_opts(echo_cmd_opts_t &opts, int *optind, int argc, wchar_t
DIE("unexpected retval from wgetopt_long"); DIE("unexpected retval from wgetopt_long");
} }
} }
// Super cheesy: We keep an old copy of the option state around,
// so we can revert it in case we get an argument like
// "-n foo".
// We need to keep it one out-of-date so we can ignore the *last* option.
// (this might be an issue in wgetopt, but that's a whole other can of worms
// and really only occurs with our weird "put it back" option parsing)
if (w.woptind == oldoptind + 2) {
oldopts = opts;
oldoptind = w.woptind;
}
} }
*optind = w.woptind; *optind = w.woptind;

View file

@ -216,8 +216,8 @@ maybe_t<int> builtin_history(parser_t &parser, io_streams_t &streams, wchar_t **
// Use the default history if we have none (which happens if invoked non-interactively, e.g. // Use the default history if we have none (which happens if invoked non-interactively, e.g.
// from webconfig.py. // from webconfig.py.
history_t *history = reader_get_history(); std::shared_ptr<history_t> history = reader_get_history();
if (!history) history = &history_t::history_with_name(history_session_id(parser.vars())); if (!history) history = history_t::with_name(history_session_id(parser.vars()));
// If a history command hasn't already been specified via a flag check the first word. // If a history command hasn't already been specified via a flag check the first word.
// Note that this can be simplified after we eliminate allowing subcommands as flags. // Note that this can be simplified after we eliminate allowing subcommands as flags.

View file

@ -483,8 +483,8 @@ static int builtin_set_list(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
if (!names_only) { if (!names_only) {
wcstring val; wcstring val;
if (opts.shorten_ok && key == L"history") { if (opts.shorten_ok && key == L"history") {
history_t *history = std::shared_ptr<history_t> history =
&history_t::history_with_name(history_session_id(parser.vars())); history_t::with_name(history_session_id(parser.vars()));
for (size_t i = 1; i < history->size() && val.size() < 64; i++) { for (size_t i = 1; i < history->size() && val.size() < 64; i++) {
if (i > 1) val += L' '; if (i > 1) val += L' ';
val += expand_escape_string(history->item_at_index(i).str()); val += expand_escape_string(history->item_at_index(i).str());

View file

@ -134,7 +134,7 @@ class arg_iterator_t {
// This is used by the string subcommands to communicate with the option parser which flags are // This is used by the string subcommands to communicate with the option parser which flags are
// valid and get the result of parsing the command for flags. // valid and get the result of parsing the command for flags.
using options_t = struct options_t { //!OCLINT(too many fields) struct options_t { //!OCLINT(too many fields)
bool all_valid = false; bool all_valid = false;
bool char_to_pad_valid = false; bool char_to_pad_valid = false;
bool chars_to_trim_valid = false; bool chars_to_trim_valid = false;
@ -1558,26 +1558,40 @@ static int string_repeat(parser_t &parser, io_streams_t &streams, int argc, wcha
int retval = parse_opts(&opts, &optind, 0, argc, argv, parser, streams); int retval = parse_opts(&opts, &optind, 0, argc, argv, parser, streams);
if (retval != STATUS_CMD_OK) return retval; if (retval != STATUS_CMD_OK) return retval;
bool is_empty = true; bool all_empty = true;
bool first = true;
arg_iterator_t aiter(argv, optind, streams); arg_iterator_t aiter(argv, optind, streams);
if (const wcstring *word = aiter.nextstr()) { while (const wcstring *word = aiter.nextstr()) {
if (!first && !opts.quiet) {
streams.out.append(L'\n');
}
first = false;
const bool limit_repeat = const bool limit_repeat =
(opts.max > 0 && word->length() * opts.count > static_cast<size_t>(opts.max)) || (opts.max > 0 && word->length() * opts.count > static_cast<size_t>(opts.max)) ||
!opts.count; !opts.count;
const wcstring repeated = const wcstring repeated =
limit_repeat ? wcsrepeat_until(*word, opts.max) : wcsrepeat(*word, opts.count); limit_repeat ? wcsrepeat_until(*word, opts.max) : wcsrepeat(*word, opts.count);
is_empty = repeated.empty(); if (!repeated.empty()) {
all_empty = false;
if (opts.quiet) {
// Early out if we can - see #7495.
return STATUS_CMD_OK;
}
}
if (!opts.quiet && !is_empty) { // Append if not quiet.
if (!opts.quiet) {
streams.out.append(repeated); streams.out.append(repeated);
if (!opts.no_newline) streams.out.append(L"\n");
} else if (opts.quiet && !is_empty) {
return STATUS_CMD_OK;
} }
} }
return !is_empty ? STATUS_CMD_OK : STATUS_CMD_ERROR; // Historical behavior is to never append a newline if all strings were empty.
if (!opts.quiet && !opts.no_newline && !all_empty) {
streams.out.append(L'\n');
}
return all_empty ? STATUS_CMD_ERROR : STATUS_CMD_OK;
} }
static int string_sub(parser_t &parser, io_streams_t &streams, int argc, wchar_t **argv) { static int string_sub(parser_t &parser, io_streams_t &streams, int argc, wchar_t **argv) {

View file

@ -155,7 +155,7 @@ maybe_t<int> builtin_type(parser_t &parser, io_streams_t &streams, wchar_t **arg
def = comment.append(def); def = comment.append(def);
} else { } else {
wcstring comment; wcstring comment;
append_format(comment, L"# Defined interactively"); append_format(comment, L"# Defined interactively\n");
def = comment.append(def); def = comment.append(def);
} }
if (!streams.out_is_redirected && isatty(STDOUT_FILENO)) { if (!streams.out_is_redirected && isatty(STDOUT_FILENO)) {

View file

@ -1252,8 +1252,8 @@ bool completer_t::complete_variable(const wcstring &str, size_t start_offset) {
// $history can be huge, don't put all of it in the completion description; see // $history can be huge, don't put all of it in the completion description; see
// #6288. // #6288.
if (env_name == L"history") { if (env_name == L"history") {
history_t *history = std::shared_ptr<history_t> history =
&history_t::history_with_name(history_session_id(ctx.vars)); history_t::with_name(history_session_id(ctx.vars));
for (size_t i = 1; i < history->size() && desc.size() < 64; i++) { for (size_t i = 1; i < history->size() && desc.size() < 64; i++) {
if (i > 1) desc += L' '; if (i > 1) desc += L' ';
desc += expand_escape_string(history->item_at_index(i).str()); desc += expand_escape_string(history->item_at_index(i).str());

View file

@ -90,7 +90,6 @@ static const std::vector<electric_var_t> electric_variables{
{L"_", electric_var_t::freadonly}, {L"_", electric_var_t::freadonly},
{L"fish_kill_signal", electric_var_t::freadonly | electric_var_t::fcomputed}, {L"fish_kill_signal", electric_var_t::freadonly | electric_var_t::fcomputed},
{L"fish_pid", electric_var_t::freadonly}, {L"fish_pid", electric_var_t::freadonly},
{L"fish_private_mode", electric_var_t::freadonly},
{L"history", electric_var_t::freadonly | electric_var_t::fcomputed}, {L"history", electric_var_t::freadonly | electric_var_t::fcomputed},
{L"hostname", electric_var_t::freadonly}, {L"hostname", electric_var_t::freadonly},
{L"pipestatus", electric_var_t::freadonly | electric_var_t::fcomputed}, {L"pipestatus", electric_var_t::freadonly | electric_var_t::fcomputed},
@ -119,8 +118,7 @@ static bool is_read_only(const wcstring &key) {
if (auto ev = electric_var_t::for_name(key)) { if (auto ev = electric_var_t::for_name(key)) {
return ev->readonly(); return ev->readonly();
} }
// Hack. return false;
return in_private_mode() && key == L"fish_history";
} }
/// Return true if a variable should become a path variable by default. See #436. /// Return true if a variable should become a path variable by default. See #436.
@ -275,17 +273,6 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
vars.set_one(FISH_BIN_DIR, ENV_GLOBAL, paths->bin); vars.set_one(FISH_BIN_DIR, ENV_GLOBAL, paths->bin);
} }
wcstring user_config_dir;
path_get_config(user_config_dir);
vars.set_one(FISH_CONFIG_DIR, ENV_GLOBAL, user_config_dir);
wcstring user_data_dir;
path_get_data(user_data_dir);
vars.set_one(FISH_USER_DATA_DIR, ENV_GLOBAL, user_data_dir);
// Set up the USER and PATH variables
setup_path();
// Some `su`s keep $USER when changing to root. // Some `su`s keep $USER when changing to root.
// This leads to issues later on (and e.g. in prompts), // This leads to issues later on (and e.g. in prompts),
// so we work around it by resetting $USER. // so we work around it by resetting $USER.
@ -294,40 +281,13 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
uid_t uid = getuid(); uid_t uid = getuid();
setup_user(uid == 0); setup_user(uid == 0);
// Set up $IFS - this used to be in share/config.fish, but really breaks if it isn't done.
vars.set_one(L"IFS", ENV_GLOBAL, L"\n \t");
// Set up the version variable.
wcstring version = str2wcstring(get_fish_version());
vars.set_one(L"version", ENV_GLOBAL, version);
vars.set_one(L"FISH_VERSION", ENV_GLOBAL, version);
// Set the $fish_pid variable.
vars.set_one(L"fish_pid", ENV_GLOBAL, to_string(getpid()));
// Set the $hostname variable
wcstring hostname = L"fish";
get_hostname_identifier(hostname);
vars.set_one(L"hostname", ENV_GLOBAL, hostname);
// Set up SHLVL variable. Not we can't use vars.get() because SHLVL is read-only, and therefore
// was not inherited from the environment.
wcstring nshlvl_str = L"1";
if (const char *shlvl_var = getenv("SHLVL")) {
const wchar_t *end;
// TODO: Figure out how to handle invalid numbers better. Shouldn't we issue a diagnostic?
long shlvl_i = fish_wcstol(str2wcstring(shlvl_var).c_str(), &end);
if (!errno && shlvl_i >= 0) {
nshlvl_str = to_string(shlvl_i + 1);
}
}
vars.set_one(L"SHLVL", ENV_GLOBAL | ENV_EXPORT, nshlvl_str);
// Set up the HOME variable. // Set up the HOME variable.
// Unlike $USER, it doesn't seem that `su`s pass this along // Unlike $USER, it doesn't seem that `su`s pass this along
// if the target user is root, unless "--preserve-environment" is used. // if the target user is root, unless "--preserve-environment" is used.
// Since that is an explicit choice, we should allow it to enable e.g. // Since that is an explicit choice, we should allow it to enable e.g.
// env HOME=(mktemp -d) su --preserve-environment fish // env HOME=(mktemp -d) su --preserve-environment fish
//
// Note: This needs to be *before* path_get_*, because that uses $HOME!
if (vars.get(L"HOME").missing_or_empty()) { if (vars.get(L"HOME").missing_or_empty()) {
auto user_var = vars.get(L"USER"); auto user_var = vars.get(L"USER");
if (!user_var.missing_or_empty()) { if (!user_var.missing_or_empty()) {
@ -359,6 +319,46 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
} }
} }
wcstring user_config_dir;
path_get_config(user_config_dir);
vars.set_one(FISH_CONFIG_DIR, ENV_GLOBAL, user_config_dir);
wcstring user_data_dir;
path_get_data(user_data_dir);
vars.set_one(FISH_USER_DATA_DIR, ENV_GLOBAL, user_data_dir);
// Set up a default PATH
setup_path();
// Set up $IFS - this used to be in share/config.fish, but really breaks if it isn't done.
vars.set_one(L"IFS", ENV_GLOBAL, L"\n \t");
// Set up the version variable.
wcstring version = str2wcstring(get_fish_version());
vars.set_one(L"version", ENV_GLOBAL, version);
vars.set_one(L"FISH_VERSION", ENV_GLOBAL, version);
// Set the $fish_pid variable.
vars.set_one(L"fish_pid", ENV_GLOBAL, to_string(getpid()));
// Set the $hostname variable
wcstring hostname = L"fish";
get_hostname_identifier(hostname);
vars.set_one(L"hostname", ENV_GLOBAL, hostname);
// Set up SHLVL variable. Not we can't use vars.get() because SHLVL is read-only, and therefore
// was not inherited from the environment.
wcstring nshlvl_str = L"1";
if (const char *shlvl_var = getenv("SHLVL")) {
const wchar_t *end;
// TODO: Figure out how to handle invalid numbers better. Shouldn't we issue a diagnostic?
long shlvl_i = fish_wcstol(str2wcstring(shlvl_var).c_str(), &end);
if (!errno && shlvl_i >= 0) {
nshlvl_str = to_string(shlvl_i + 1);
}
}
vars.set_one(L"SHLVL", ENV_GLOBAL | ENV_EXPORT, nshlvl_str);
// initialize the PWD variable if necessary // initialize the PWD variable if necessary
// Note we may inherit a virtual PWD that doesn't match what getcwd would return; respect that // Note we may inherit a virtual PWD that doesn't match what getcwd would return; respect that
// if and only if it matches getcwd (#5647). Note we treat PWD as read-only so it was not set in // if and only if it matches getcwd (#5647). Note we treat PWD as read-only so it was not set in
@ -689,9 +689,9 @@ maybe_t<env_var_t> env_scoped_impl_t::try_get_computed(const wcstring &key) cons
return none(); return none();
} }
history_t *history = reader_get_history(); std::shared_ptr<history_t> history = reader_get_history();
if (!history) { if (!history) {
history = &history_t::history_with_name(history_session_id(*this)); history = history_t::with_name(history_session_id(*this));
} }
wcstring_list_t result; wcstring_list_t result;
if (history) history->get_history(result); if (history) history->get_history(result);

View file

@ -535,7 +535,7 @@ autoclose_fd_t env_universal_t::open_temporary_file(const wcstring &directory, w
// This should almost always succeed on the first try. // This should almost always succeed on the first try.
assert(!string_suffixes_string(L"/", directory)); //!OCLINT(multiple unary operator) assert(!string_suffixes_string(L"/", directory)); //!OCLINT(multiple unary operator)
int saved_errno; int saved_errno = 0;
const wcstring tmp_name_template = directory + L"/fishd.tmp.XXXXXX"; const wcstring tmp_name_template = directory + L"/fishd.tmp.XXXXXX";
autoclose_fd_t result; autoclose_fd_t result;
std::string narrow_str; std::string narrow_str;

View file

@ -355,13 +355,13 @@ static expand_result_t expand_variables(wcstring instr, completion_receiver_t *o
// Do a dirty hack to make sliced history fast (#4650). We expand from either a variable, or a // Do a dirty hack to make sliced history fast (#4650). We expand from either a variable, or a
// history_t. Note that "history" is read only in env.cpp so it's safe to special-case it in // history_t. Note that "history" is read only in env.cpp so it's safe to special-case it in
// this way (it cannot be shadowed, etc). // this way (it cannot be shadowed, etc).
history_t *history = nullptr; std::shared_ptr<history_t> history{};
maybe_t<env_var_t> var{}; maybe_t<env_var_t> var{};
if (var_name == L"history") { if (var_name == L"history") {
// Note reader_get_history may return null, if we are running non-interactively (e.g. from // Note reader_get_history may return null, if we are running non-interactively (e.g. from
// web_config). // web_config).
if (is_main_thread()) { if (is_main_thread()) {
history = &history_t::history_with_name(history_session_id(env_stack_t::principal())); history = history_t::with_name(history_session_id(env_stack_t::principal()));
} }
} else if (var_name != wcstring{VARIABLE_EXPAND_EMPTY}) { } else if (var_name != wcstring{VARIABLE_EXPAND_EMPTY}) {
var = vars.get(var_name); var = vars.get(var_name);

View file

@ -26,15 +26,16 @@ fd_monitor_t::fd_monitor_t() {
// Add an item for ourselves. // Add an item for ourselves.
// We don't need to go through 'pending' because we have not yet launched the thread, and don't // We don't need to go through 'pending' because we have not yet launched the thread, and don't
// want to yet. // want to yet.
auto callback = [this](const autoclose_fd_t &fd, bool timed_out) { auto callback = [this](const autoclose_fd_t &fd, item_wake_reason_t reason) {
ASSERT_IS_BACKGROUND_THREAD(); ASSERT_IS_BACKGROUND_THREAD();
assert(!timed_out && "Should not time out with kNoTimeout"); assert(reason == item_wake_reason_t::readable &&
(void)timed_out; "Should not be poked, or time out with kNoTimeout");
(void)reason;
// Read some to take data off of the notifier. // Read some to take data off of the notifier.
char buff[4096]; char buff[4096];
ssize_t amt = read(fd.fd(), buff, sizeof buff); ssize_t amt = read(fd.fd(), buff, sizeof buff);
if (amt > 0) { if (amt > 0) {
this->has_pending_ = true; this->has_pending_or_pokes_ = true;
} else if (amt == 0) { } else if (amt == 0) {
this->terminate_ = true; this->terminate_ = true;
} else { } else {
@ -54,10 +55,28 @@ fd_monitor_t::~fd_monitor_t() {
} }
} }
void fd_monitor_t::add(fd_monitor_item_t &&item) { fd_monitor_item_id_t fd_monitor_t::add(fd_monitor_item_t &&item) {
assert(item.fd.valid() && "Invalid fd"); assert(item.fd.valid() && "Invalid fd");
assert(item.timeout_usec != 0 && "Invalid timeout"); assert(item.timeout_usec != 0 && "Invalid timeout");
bool start_thread = add_pending_get_start_thread(std::move(item)); assert(item.item_id == 0 && "Item should not already have an ID");
bool start_thread = false;
fd_monitor_item_id_t item_id{};
{
// Lock around a local region.
auto data = data_.acquire();
// Assign an id and add the item to pending.
item_id = ++data->last_id;
item.item_id = item_id;
data->pending.push_back(std::move(item));
// Maybe plan to start the thread.
if (!data->running) {
FLOG(fd_monitor, "Thread starting");
data->running = true;
start_thread = true;
}
}
if (start_thread) { if (start_thread) {
void *(*trampoline)(void *) = [](void *self) -> void * { void *(*trampoline)(void *) = [](void *self) -> void * {
static_cast<fd_monitor_t *>(self)->run_in_background(); static_cast<fd_monitor_t *>(self)->run_in_background();
@ -71,17 +90,24 @@ void fd_monitor_t::add(fd_monitor_item_t &&item) {
// Tickle our notifier. // Tickle our notifier.
char byte = 0; char byte = 0;
(void)write_loop(notify_write_fd_.fd(), &byte, 1); (void)write_loop(notify_write_fd_.fd(), &byte, 1);
return item_id;
} }
bool fd_monitor_t::add_pending_get_start_thread(fd_monitor_item_t &&item) { void fd_monitor_t::poke_item(fd_monitor_item_id_t item_id) {
auto data = data_.acquire(); assert(item_id > 0 && "Invalid item ID");
data->pending.push_back(std::move(item)); bool needs_notifier_byte = false;
if (!data->running) { {
FLOG(fd_monitor, "Thread starting"); auto data = data_.acquire();
data->running = true; needs_notifier_byte = data->pokelist.empty();
return true; // Insert it, sorted.
auto where = std::lower_bound(data->pokelist.begin(), data->pokelist.end(), item_id);
data->pokelist.insert(where, item_id);
}
if (needs_notifier_byte) {
// Tickle our notifier.
char byte = 0;
(void)write_loop(notify_write_fd_.fd(), &byte, 1);
} }
return false;
} }
// Given a usec count, populate and return a timeval. // Given a usec count, populate and return a timeval.
@ -108,15 +134,33 @@ bool fd_monitor_item_t::service_item(const fd_set *fds, const time_point_t &now)
bool timed_out = !readable && usec_remaining(now) == 0; bool timed_out = !readable && usec_remaining(now) == 0;
if (readable || timed_out) { if (readable || timed_out) {
last_time = now; last_time = now;
callback(fd, timed_out); item_wake_reason_t reason =
readable ? item_wake_reason_t::readable : item_wake_reason_t::timeout;
callback(fd, reason);
should_retain = fd.valid(); should_retain = fd.valid();
} }
return should_retain; return should_retain;
} }
bool fd_monitor_item_t::poke_item(const poke_list_t &pokelist) {
if (item_id == 0 || !std::binary_search(pokelist.begin(), pokelist.end(), item_id)) {
// Not pokeable or not in the pokelist.
return true;
}
callback(fd, item_wake_reason_t::poke);
return fd.valid();
}
void fd_monitor_t::run_in_background() { void fd_monitor_t::run_in_background() {
ASSERT_IS_BACKGROUND_THREAD(); ASSERT_IS_BACKGROUND_THREAD();
poke_list_t pokelist;
for (;;) { for (;;) {
// Poke any items that need it.
if (!pokelist.empty()) {
this->poke_in_background(std::move(pokelist));
pokelist.clear();
}
uint64_t timeout_usec = fd_monitor_item_t::kNoTimeout; uint64_t timeout_usec = fd_monitor_item_t::kNoTimeout;
int max_fd = -1; int max_fd = -1;
fd_set fds; fd_set fds;
@ -158,7 +202,7 @@ void fd_monitor_t::run_in_background() {
return remove; return remove;
}; };
// Service all items that are either readable or timed our, and remove any which say to do // Service all items that are either readable or timed out, and remove any which say to do
// so. // so.
now = std::chrono::steady_clock::now(); now = std::chrono::steady_clock::now();
items_.erase(std::remove_if(items_.begin(), items_.end(), servicer), items_.end()); items_.erase(std::remove_if(items_.begin(), items_.end(), servicer), items_.end());
@ -171,13 +215,19 @@ void fd_monitor_t::run_in_background() {
// Maybe we got some new items. Check if our callback says so, or if this is the wait // Maybe we got some new items. Check if our callback says so, or if this is the wait
// lap, in which case we might want to commit to exiting. // lap, in which case we might want to commit to exiting.
if (has_pending_ || is_wait_lap) { if (has_pending_or_pokes_ || is_wait_lap) {
auto data = data_.acquire(); auto data = data_.acquire();
// Move from 'pending' to 'items'. // Move from 'pending' to 'items'.
items_.insert(items_.end(), std::make_move_iterator(data->pending.begin()), items_.insert(items_.end(), std::make_move_iterator(data->pending.begin()),
std::make_move_iterator(data->pending.end())); std::make_move_iterator(data->pending.end()));
data->pending.clear(); data->pending.clear();
has_pending_ = false;
// Grab any pokelist.
assert(pokelist.empty() && "pokelist should be empty or else we're dropping pokes");
pokelist = std::move(data->pokelist);
data->pokelist.clear();
has_pending_or_pokes_ = false;
if (is_wait_lap && items_.size() == 1) { if (is_wait_lap && items_.size() == 1) {
// We had no items, waited a bit, and still have no items. We're going to shut down. // We had no items, waited a bit, and still have no items. We're going to shut down.
@ -191,3 +241,14 @@ void fd_monitor_t::run_in_background() {
} }
} }
} }
void fd_monitor_t::poke_in_background(const poke_list_t &pokelist) {
ASSERT_IS_BACKGROUND_THREAD();
auto poker = [&pokelist](fd_monitor_item_t &item) {
int fd = item.fd.fd();
bool remove = !item.poke_item(pokelist);
if (remove) FLOG(fd_monitor, "Removing fd", fd);
return remove;
};
items_.erase(std::remove_if(items_.begin(), items_.end(), poker), items_.end());
}

View file

@ -12,14 +12,24 @@
class fd_monitor_t; class fd_monitor_t;
/// Each item added to fd_monitor_t is assigned a unique ID, which is not recycled.
/// Items may have their callback triggered immediately by passing the ID.
/// Zero is a sentinel.
using fd_monitor_item_id_t = uint64_t;
/// Reasons for waking an item.
enum class item_wake_reason_t {
readable, // the fd became readable
timeout, // the requested timeout was hit
poke, // the item was "poked" (woken up explicitly)
};
/// An item containing an fd and callback, which can be monitored to watch when it becomes readable, /// An item containing an fd and callback, which can be monitored to watch when it becomes readable,
/// and invoke the callback. /// and invoke the callback.
struct fd_monitor_item_t { struct fd_monitor_item_t {
friend class fd_monitor_t; /// The callback type for the item. It is passed \p fd, and the reason for waking \p reason.
/// The callback may close \p fd, in which case the item is removed.
/// The callback type for the item. using callback_t = std::function<void(autoclose_fd_t &fd, item_wake_reason_t reason)>;
/// It will be invoked when either \p fd is readable, or if the timeout was hit.
using callback_t = std::function<void(autoclose_fd_t &fd, bool timed_out)>;
/// A sentinel value meaning no timeout. /// A sentinel value meaning no timeout.
static constexpr uint64_t kNoTimeout = std::numeric_limits<uint64_t>::max(); static constexpr uint64_t kNoTimeout = std::numeric_limits<uint64_t>::max();
@ -51,6 +61,9 @@ struct fd_monitor_item_t {
// The last time we were called, or the initialization point. // The last time we were called, or the initialization point.
maybe_t<time_point_t> last_time{}; maybe_t<time_point_t> last_time{};
// The ID for this item. This is assigned by the fd monitor.
fd_monitor_item_id_t item_id{0};
// \return the number of microseconds until the timeout should trigger, or kNoTimeout for none. // \return the number of microseconds until the timeout should trigger, or kNoTimeout for none.
// A 0 return means we are at or past the timeout. // A 0 return means we are at or past the timeout.
uint64_t usec_remaining(const time_point_t &now) const; uint64_t usec_remaining(const time_point_t &now) const;
@ -58,6 +71,13 @@ struct fd_monitor_item_t {
// Invoke this item's callback if its value is set in fd or has timed out. // Invoke this item's callback if its value is set in fd or has timed out.
// \return true to retain the item, false to remove it. // \return true to retain the item, false to remove it.
bool service_item(const fd_set *fds, const time_point_t &now); bool service_item(const fd_set *fds, const time_point_t &now);
// Invoke this item's callback with a poke, if its ID is present in the (sorted) pokelist.
// \return true to retain the item, false to remove it.
using poke_list_t = std::vector<fd_monitor_item_id_t>;
bool poke_item(const poke_list_t &pokelist);
friend class fd_monitor_t;
}; };
/// A class which can monitor a set of fds, invoking a callback when any becomes readable, or when /// A class which can monitor a set of fds, invoking a callback when any becomes readable, or when
@ -66,34 +86,47 @@ class fd_monitor_t {
public: public:
using item_list_t = std::vector<fd_monitor_item_t>; using item_list_t = std::vector<fd_monitor_item_t>;
// A "pokelist" is a sorted list of item IDs which need explicit wakeups.
using poke_list_t = std::vector<fd_monitor_item_id_t>;
fd_monitor_t(); fd_monitor_t();
~fd_monitor_t(); ~fd_monitor_t();
/// Add an item to monitor. /// Add an item to monitor. \return the ID assigned to the item.
void add(fd_monitor_item_t &&item); fd_monitor_item_id_t add(fd_monitor_item_t &&item);
/// Mark that an item with a given ID needs to be explicitly woken up.
void poke_item(fd_monitor_item_id_t item_id);
private: private:
// The background thread runner. // The background thread runner.
void run_in_background(); void run_in_background();
// Add a pending item, marking the thread as running. // Poke items in the pokelist, removing any items that close their FD.
// \return true if we should start the thread. // The pokelist is consumed after this.
bool add_pending_get_start_thread(fd_monitor_item_t &&item); // This is only called in the background thread.
void poke_in_background(const poke_list_t &pokelist);
// The list of items to monitor. This is only accessed on the background thread. // The list of items to monitor. This is only accessed on the background thread.
item_list_t items_{}; item_list_t items_{};
// Set to true by the background thread when our self-pipe becomes readable. // Set to true by the background thread when our self-pipe becomes readable.
bool has_pending_{false}; bool has_pending_or_pokes_{false};
// Latched to true by the background thread if our self-pipe is closed, which indicates we are // Latched to true by the background thread if our self-pipe is closed, which indicates we are
// in the destructor and so should terminate. // in the destructor and so should terminate.
bool terminate_{false}; bool terminate_{false};
struct data_t { struct data_t {
/// Pending items. /// Pending items. This is set under the lock, then the background thread grabs them.
item_list_t pending{}; item_list_t pending{};
/// List of IDs for items that need to be poked (explicitly woken up).
poke_list_t pokelist{};
/// The last ID assigned, or if none.
fd_monitor_item_id_t last_id{0};
/// Whether the thread is running. /// Whether the thread is running.
bool running{false}; bool running{false};
}; };

View file

@ -503,6 +503,10 @@ int main(int argc, char **argv) {
parser.libdata().exit_current_script = false; parser.libdata().exit_current_script = false;
} else if (my_optind == argc) { } else if (my_optind == argc) {
// Implicitly interactive mode. // Implicitly interactive mode.
if (opts.no_exec && isatty(STDIN_FILENO)) {
FLOGF(error, L"no-execute mode enabled and no script given. Exiting");
return EXIT_FAILURE; // above line should always exit
}
res = reader_read(parser, STDIN_FILENO, {}); res = reader_read(parser, STDIN_FILENO, {});
} else { } else {
const char *file = *(argv + (my_optind++)); const char *file = *(argv + (my_optind++));

View file

@ -787,7 +787,9 @@ static void test_fd_monitor() {
struct item_maker_t { struct item_maker_t {
std::atomic<bool> did_timeout{false}; std::atomic<bool> did_timeout{false};
std::atomic<size_t> length_read{0}; std::atomic<size_t> length_read{0};
std::atomic<size_t> pokes{0};
std::atomic<size_t> total_calls{0}; std::atomic<size_t> total_calls{0};
fd_monitor_item_id_t item_id{0};
bool always_exit{false}; bool always_exit{false};
fd_monitor_item_t item; fd_monitor_item_t item;
autoclose_fd_t writer; autoclose_fd_t writer;
@ -795,15 +797,21 @@ static void test_fd_monitor() {
explicit item_maker_t(uint64_t timeout_usec) { explicit item_maker_t(uint64_t timeout_usec) {
auto pipes = make_autoclose_pipes({}).acquire(); auto pipes = make_autoclose_pipes({}).acquire();
writer = std::move(pipes.write); writer = std::move(pipes.write);
auto callback = [this](autoclose_fd_t &fd, bool timed_out) { auto callback = [this](autoclose_fd_t &fd, item_wake_reason_t reason) {
bool was_closed = false; bool was_closed = false;
if (timed_out) { switch (reason) {
this->did_timeout = true; case item_wake_reason_t::timeout:
} else { this->did_timeout = true;
char buff[4096]; break;
ssize_t amt = read(fd.fd(), buff, sizeof buff); case item_wake_reason_t::poke:
length_read += amt; this->pokes += 1;
was_closed = (amt == 0); break;
case item_wake_reason_t::readable:
char buff[4096];
ssize_t amt = read(fd.fd(), buff, sizeof buff);
this->length_read += amt;
was_closed = (amt == 0);
break;
} }
total_calls += 1; total_calls += 1;
if (always_exit || was_closed) { if (always_exit || was_closed) {
@ -816,7 +824,7 @@ static void test_fd_monitor() {
item_maker_t(const item_maker_t &) = delete; item_maker_t(const item_maker_t &) = delete;
// Write 42 bytes to our write end. // Write 42 bytes to our write end.
void write42() { void write42() const {
char buff[42] = {0}; char buff[42] = {0};
(void)write_loop(writer.fd(), buff, sizeof buff); (void)write_loop(writer.fd(), buff, sizeof buff);
} }
@ -826,7 +834,7 @@ static void test_fd_monitor() {
// Items which will never receive data or be called back. // Items which will never receive data or be called back.
item_maker_t item_never(fd_monitor_item_t::kNoTimeout); item_maker_t item_never(fd_monitor_item_t::kNoTimeout);
item_maker_t item_hugetimeout(100000000llu * usec_per_msec); item_maker_t item_hugetimeout(100000000LLU * usec_per_msec);
// Item which should get no data, and time out. // Item which should get no data, and time out.
item_maker_t item0_timeout(16 * usec_per_msec); item_maker_t item0_timeout(16 * usec_per_msec);
@ -840,45 +848,63 @@ static void test_fd_monitor() {
// Item which should get 42 bytes, then get notified it is closed. // Item which should get 42 bytes, then get notified it is closed.
item_maker_t item42_thenclose(16 * usec_per_msec); item_maker_t item42_thenclose(16 * usec_per_msec);
// Item which gets one poke.
item_maker_t item_pokee(fd_monitor_item_t::kNoTimeout);
// Item which should be called back once. // Item which should be called back once.
item_maker_t item_oneshot(16 * usec_per_msec); item_maker_t item_oneshot(16 * usec_per_msec);
item_oneshot.always_exit = true; item_oneshot.always_exit = true;
{ {
fd_monitor_t monitor; fd_monitor_t monitor;
for (auto item : {&item_never, &item_hugetimeout, &item0_timeout, &item42_timeout, for (item_maker_t *item :
&item42_nottimeout, &item42_thenclose, &item_oneshot}) { {&item_never, &item_hugetimeout, &item0_timeout, &item42_timeout, &item42_nottimeout,
monitor.add(std::move(item->item)); &item42_thenclose, &item_pokee, &item_oneshot}) {
item->item_id = monitor.add(std::move(item->item));
} }
item42_timeout.write42(); item42_timeout.write42();
item42_nottimeout.write42(); item42_nottimeout.write42();
item42_thenclose.write42(); item42_thenclose.write42();
item42_thenclose.writer.close(); item42_thenclose.writer.close();
item_oneshot.write42(); item_oneshot.write42();
monitor.poke_item(item_pokee.item_id);
std::this_thread::sleep_for(std::chrono::milliseconds(84)); std::this_thread::sleep_for(std::chrono::milliseconds(84));
} }
do_test(!item_never.did_timeout); do_test(!item_never.did_timeout);
do_test(item_never.length_read == 0); do_test(item_never.length_read == 0);
do_test(item_never.pokes == 0);
do_test(!item_hugetimeout.did_timeout); do_test(!item_hugetimeout.did_timeout);
do_test(item_hugetimeout.length_read == 0); do_test(item_hugetimeout.length_read == 0);
do_test(item_hugetimeout.pokes == 0);
do_test(item0_timeout.length_read == 0); do_test(item0_timeout.length_read == 0);
do_test(item0_timeout.did_timeout); do_test(item0_timeout.did_timeout);
do_test(item0_timeout.pokes == 0);
do_test(item42_timeout.length_read == 42); do_test(item42_timeout.length_read == 42);
do_test(item42_timeout.did_timeout); do_test(item42_timeout.did_timeout);
do_test(item42_timeout.pokes == 0);
do_test(item42_nottimeout.length_read == 42); do_test(item42_nottimeout.length_read == 42);
do_test(!item42_nottimeout.did_timeout); do_test(!item42_nottimeout.did_timeout);
do_test(item42_nottimeout.pokes == 0);
do_test(item42_thenclose.did_timeout == false); do_test(item42_thenclose.did_timeout == false);
do_test(item42_thenclose.length_read == 42); do_test(item42_thenclose.length_read == 42);
do_test(item42_thenclose.total_calls == 2); do_test(item42_thenclose.total_calls == 2);
do_test(item42_thenclose.pokes == 0);
do_test(!item_oneshot.did_timeout); do_test(!item_oneshot.did_timeout);
do_test(item_oneshot.length_read == 42); do_test(item_oneshot.length_read == 42);
do_test(item_oneshot.total_calls == 1); do_test(item_oneshot.total_calls == 1);
do_test(item_oneshot.pokes == 0);
do_test(!item_pokee.did_timeout);
do_test(item_pokee.length_read == 0);
do_test(item_pokee.total_calls == 1);
do_test(item_pokee.pokes == 1);
} }
static void test_iothread() { static void test_iothread() {
@ -1269,7 +1295,7 @@ static void test_parser() {
static void test_1_cancellation(const wchar_t *src) { static void test_1_cancellation(const wchar_t *src) {
auto filler = io_bufferfill_t::create(fd_set_t{}); auto filler = io_bufferfill_t::create(fd_set_t{});
pthread_t thread = pthread_self(); pthread_t thread = pthread_self();
double delay = 0.25 /* seconds */; double delay = 0.50 /* seconds */;
iothread_perform([=]() { iothread_perform([=]() {
/// Wait a while and then SIGINT the main thread. /// Wait a while and then SIGINT the main thread.
usleep(delay * 1E6); usleep(delay * 1E6);
@ -1616,7 +1642,8 @@ static void test_wchar2utf8(const wchar_t *src, size_t slen, const char *dst, si
#endif #endif
if (dst) { if (dst) {
mem = (char *)malloc(dlen); // We want to pass a valid pointer to wchar_to_utf8, so allocate at least one byte.
mem = (char *)malloc(dlen + 1);
if (!mem) { if (!mem) {
err(L"w2u: %s: MALLOC FAILED", descr); err(L"w2u: %s: MALLOC FAILED", descr);
return; return;
@ -1883,26 +1910,47 @@ static void test_lru() {
do_test(cache.evicted.size() == size_t(total_nodes)); do_test(cache.evicted.size() == size_t(total_nodes));
} }
/// A crappy environment_t that only knows about PWD. /// An environment built around an std::map.
struct pwd_environment_t : public environment_t { struct test_environment_t : public environment_t {
std::map<wcstring, wcstring> extras; std::map<wcstring, wcstring> vars;
virtual maybe_t<env_var_t> get(const wcstring &key, virtual maybe_t<env_var_t> get(const wcstring &key,
env_mode_flags_t mode = ENV_DEFAULT) const override { env_mode_flags_t mode = ENV_DEFAULT) const override {
UNUSED(mode); UNUSED(mode);
if (key == L"PWD") { auto iter = vars.find(key);
return env_var_t{wgetcwd(), 0}; if (iter != vars.end()) {
return env_var_t(iter->second, ENV_DEFAULT);
} }
auto extra = extras.find(key); return none();
if (extra != extras.end()) {
return env_var_t(extra->second, ENV_DEFAULT);
}
return {};
} }
wcstring_list_t get_names(int flags) const override { wcstring_list_t get_names(int flags) const override {
UNUSED(flags); UNUSED(flags);
return {L"PWD"}; wcstring_list_t result;
for (const auto &kv : vars) {
result.push_back(kv.first);
}
return result;
}
};
/// A test environment that knows about PWD.
struct pwd_environment_t : public test_environment_t {
virtual maybe_t<env_var_t> get(const wcstring &key,
env_mode_flags_t mode = ENV_DEFAULT) const override {
if (key == L"PWD") {
return env_var_t{wgetcwd(), 0};
}
return test_environment_t::get(key, mode);
}
wcstring_list_t get_names(int flags) const override {
auto res = test_environment_t::get_names(flags);
res.clear();
if (std::count(res.begin(), res.end(), L"PWD") == 0) {
res.push_back(L"PWD");
}
return res;
} }
}; };
@ -2424,7 +2472,7 @@ struct pager_layout_testcase_t {
text.push_back(p.character); text.push_back(p.character);
} }
if (text != expected) { if (text != expected) {
std::fwprintf(stderr, L"width %zu got %zu<%ls>, expected %zu<%ls>\n", this->width, std::fwprintf(stderr, L"width %d got %zu<%ls>, expected %zu<%ls>\n", this->width,
text.length(), text.c_str(), expected.length(), expected.c_str()); text.length(), text.c_str(), expected.length(), expected.c_str());
for (size_t i = 0; i < std::max(text.length(), expected.length()); i++) { for (size_t i = 0; i < std::max(text.length(), expected.length()); i++) {
std::fwprintf(stderr, L"i %zu got <%lx> expected <%lx>\n", i, std::fwprintf(stderr, L"i %zu got <%lx> expected <%lx>\n", i,
@ -3322,7 +3370,7 @@ static void test_autosuggest_suggest_special() {
const wcstring wd = L"test/autosuggest_test"; const wcstring wd = L"test/autosuggest_test";
pwd_environment_t vars{}; pwd_environment_t vars{};
vars.extras[L"HOME"] = parser_t::principal_parser().vars().get(L"HOME")->as_string(); vars.vars[L"HOME"] = parser_t::principal_parser().vars().get(L"HOME")->as_string();
perform_one_autosuggestion_cd_test(L"cd test/autosuggest_test/0", L"foobar/", vars, __LINE__); perform_one_autosuggestion_cd_test(L"cd test/autosuggest_test/0", L"foobar/", vars, __LINE__);
perform_one_autosuggestion_cd_test(L"cd \"test/autosuggest_test/0", L"foobar/", vars, __LINE__); perform_one_autosuggestion_cd_test(L"cd \"test/autosuggest_test/0", L"foobar/", vars, __LINE__);
@ -3351,7 +3399,7 @@ static void test_autosuggest_suggest_special() {
perform_one_autosuggestion_cd_test(L"cd 'test/autosuggest_test/5", L"foo\"bar/", vars, perform_one_autosuggestion_cd_test(L"cd 'test/autosuggest_test/5", L"foo\"bar/", vars,
__LINE__); __LINE__);
vars.extras[L"AUTOSUGGEST_TEST_LOC"] = wd; vars.vars[L"AUTOSUGGEST_TEST_LOC"] = wd;
perform_one_autosuggestion_cd_test(L"cd $AUTOSUGGEST_TEST_LOC/0", L"foobar/", vars, __LINE__); perform_one_autosuggestion_cd_test(L"cd $AUTOSUGGEST_TEST_LOC/0", L"foobar/", vars, __LINE__);
perform_one_autosuggestion_cd_test(L"cd ~/test_autosuggest_suggest_specia", L"l/", vars, perform_one_autosuggestion_cd_test(L"cd ~/test_autosuggest_suggest_specia", L"l/", vars,
__LINE__); __LINE__);
@ -3460,7 +3508,7 @@ static bool history_contains(history_t *history, const wcstring &txt) {
return result; return result;
} }
static bool history_contains(const std::unique_ptr<history_t> &history, const wcstring &txt) { static bool history_contains(const std::shared_ptr<history_t> &history, const wcstring &txt) {
return history_contains(history.get(), txt); return history_contains(history.get(), txt);
} }
@ -3933,10 +3981,11 @@ class history_tests_t {
public: public:
static void test_history(); static void test_history();
static void test_history_merge(); static void test_history_merge();
static void test_history_path_detection();
static void test_history_formats(); static void test_history_formats();
// static void test_history_speed(void); // static void test_history_speed(void);
static void test_history_races(); static void test_history_races();
static void test_history_races_pound_on_history(size_t item_count); static void test_history_races_pound_on_history(size_t item_count, size_t idx);
}; };
static wcstring random_string() { static wcstring random_string() {
@ -3958,10 +4007,10 @@ void history_tests_t::test_history() {
const history_search_flags_t nocase = history_search_ignore_case; const history_search_flags_t nocase = history_search_ignore_case;
// Populate a history. // Populate a history.
history_t &history = history_t::history_with_name(L"test_history"); std::shared_ptr<history_t> history = history_t::with_name(L"test_history");
history.clear(); history->clear();
for (const wcstring &s : items) { for (const wcstring &s : items) {
history.add(s); history->add(s);
} }
// Helper to set expected items to those matching a predicate, in reverse order. // Helper to set expected items to those matching a predicate, in reverse order.
@ -4013,13 +4062,13 @@ void history_tests_t::test_history() {
// Test item removal case-sensitive. // Test item removal case-sensitive.
searcher = history_search_t(history, L"Alpha"); searcher = history_search_t(history, L"Alpha");
test_history_matches(searcher, {L"Alpha"}, __LINE__); test_history_matches(searcher, {L"Alpha"}, __LINE__);
history.remove(L"Alpha"); history->remove(L"Alpha");
searcher = history_search_t(history, L"Alpha"); searcher = history_search_t(history, L"Alpha");
test_history_matches(searcher, {}, __LINE__); test_history_matches(searcher, {}, __LINE__);
// Test history escaping and unescaping, yaml, etc. // Test history escaping and unescaping, yaml, etc.
history_item_list_t before, after; history_item_list_t before, after;
history.clear(); history->clear();
size_t i, max = 100; size_t i, max = 100;
for (i = 1; i <= max; i++) { for (i = 1; i <= max; i++) {
// Generate a value. // Generate a value.
@ -4039,17 +4088,17 @@ void history_tests_t::test_history() {
history_item_t item(value, time(NULL)); history_item_t item(value, time(NULL));
item.required_paths = paths; item.required_paths = paths;
before.push_back(item); before.push_back(item);
history.add(item); history->add(item);
} }
history.save(); history->save();
// Empty items should just be dropped (#6032). // Empty items should just be dropped (#6032).
history.add(L""); history->add(L"");
do_test(!history.item_at_index(1).contents.empty()); do_test(!history->item_at_index(1).contents.empty());
// Read items back in reverse order and ensure they're the same. // Read items back in reverse order and ensure they're the same.
for (i = 100; i >= 1; i--) { for (i = 100; i >= 1; i--) {
history_item_t item = history.item_at_index(i); history_item_t item = history->item_at_index(i);
do_test(!item.empty()); do_test(!item.empty());
after.push_back(item); after.push_back(item);
} }
@ -4062,9 +4111,8 @@ void history_tests_t::test_history() {
} }
// Clean up after our tests. // Clean up after our tests.
history.clear(); history->clear();
} }
// Wait until the next second. // Wait until the next second.
static void time_barrier() { static void time_barrier() {
time_t start = time(NULL); time_t start = time(NULL);
@ -4073,20 +4121,19 @@ static void time_barrier() {
} while (time(NULL) == start); } while (time(NULL) == start);
} }
static wcstring_list_t generate_history_lines(size_t item_count, int pid) { static wcstring_list_t generate_history_lines(size_t item_count, size_t idx) {
wcstring_list_t result; wcstring_list_t result;
result.reserve(item_count); result.reserve(item_count);
for (unsigned long i = 0; i < item_count; i++) { for (unsigned long i = 0; i < item_count; i++) {
result.push_back(format_string(L"%ld %lu", (long)pid, (unsigned long)i)); result.push_back(format_string(L"%ld %lu", (unsigned long)idx, (unsigned long)i));
} }
return result; return result;
} }
void history_tests_t::test_history_races_pound_on_history(size_t item_count) { void history_tests_t::test_history_races_pound_on_history(size_t item_count, size_t idx) {
// Called in child process to modify history. // Called in child thread to modify history.
history_t hist(L"race_test"); history_t hist(L"race_test");
hist.chaos_mode = !true; const wcstring_list_t hist_lines = generate_history_lines(item_count, idx);
const wcstring_list_t hist_lines = generate_history_lines(item_count, getpid());
for (const wcstring &line : hist_lines) { for (const wcstring &line : hist_lines) {
hist.add(line); hist.add(line);
hist.save(); hist.save();
@ -4096,6 +4143,21 @@ void history_tests_t::test_history_races_pound_on_history(size_t item_count) {
void history_tests_t::test_history_races() { void history_tests_t::test_history_races() {
say(L"Testing history race conditions"); say(L"Testing history race conditions");
// It appears TSAN and ASAN's allocators do not release their locks properly in atfork, so
// allocating with multiple threads risks deadlock. Drain threads before running under ASAN.
// TODO: stop forking with these tests.
bool needs_thread_drain = false;
#if __SANITIZE_ADDRESS__
needs_thread_drain |= true;
#endif
#if defined(__has_feature)
needs_thread_drain |= __has_feature(thread_sanitizer) || __has_feature(address_sanitizer);
#endif
if (needs_thread_drain) {
iothread_drain_all();
}
// Test concurrent history writing. // Test concurrent history writing.
// How many concurrent writers we have // How many concurrent writers we have
constexpr size_t RACE_COUNT = 4; constexpr size_t RACE_COUNT = 4;
@ -4106,30 +4168,22 @@ void history_tests_t::test_history_races() {
// Ensure history is clear. // Ensure history is clear.
history_t(L"race_test").clear(); history_t(L"race_test").clear();
pid_t children[RACE_COUNT]; // hist.chaos_mode = true;
std::thread children[RACE_COUNT];
for (size_t i = 0; i < RACE_COUNT; i++) { for (size_t i = 0; i < RACE_COUNT; i++) {
pid_t pid = fork(); children[i] = std::thread([=] { test_history_races_pound_on_history(ITEM_COUNT, i); });
if (!pid) {
// Child process.
setup_fork_guards();
test_history_races_pound_on_history(ITEM_COUNT);
exit_without_destructors(0);
} else {
// Parent process.
children[i] = pid;
}
} }
// Wait for all children. // Wait for all children.
for (pid_t child : children) { for (std::thread &child : children) {
int stat; child.join();
waitpid(child, &stat, WUNTRACED);
} }
// Compute the expected lines. // Compute the expected lines.
std::array<wcstring_list_t, RACE_COUNT> expected_lines; std::array<wcstring_list_t, RACE_COUNT> expected_lines;
for (size_t i = 0; i < RACE_COUNT; i++) { for (size_t i = 0; i < RACE_COUNT; i++) {
expected_lines[i] = generate_history_lines(ITEM_COUNT, children[i]); expected_lines[i] = generate_history_lines(ITEM_COUNT, i);
} }
// Ensure we consider the lines that have been outputted as part of our history. // Ensure we consider the lines that have been outputted as part of our history.
@ -4196,8 +4250,9 @@ void history_tests_t::test_history_merge() {
say(L"Testing history merge"); say(L"Testing history merge");
const size_t count = 3; const size_t count = 3;
const wcstring name = L"merge_test"; const wcstring name = L"merge_test";
std::unique_ptr<history_t> hists[count] = { std::shared_ptr<history_t> hists[count] = {std::make_shared<history_t>(name),
make_unique<history_t>(name), make_unique<history_t>(name), make_unique<history_t>(name)}; std::make_shared<history_t>(name),
std::make_shared<history_t>(name)};
const wcstring texts[count] = {L"History 1", L"History 2", L"History 3"}; const wcstring texts[count] = {L"History 1", L"History 2", L"History 3"};
const wcstring alt_texts[count] = {L"History Alt 1", L"History Alt 2", L"History Alt 3"}; const wcstring alt_texts[count] = {L"History Alt 1", L"History Alt 2", L"History Alt 3"};
@ -4231,7 +4286,7 @@ void history_tests_t::test_history_merge() {
// Make a new history. It should contain everything. The time_barrier() is so that the timestamp // Make a new history. It should contain everything. The time_barrier() is so that the timestamp
// is newer, since we only pick up items whose timestamp is before the birth stamp. // is newer, since we only pick up items whose timestamp is before the birth stamp.
time_barrier(); time_barrier();
std::unique_ptr<history_t> everything = make_unique<history_t>(name); std::shared_ptr<history_t> everything = std::make_shared<history_t>(name);
for (const auto &text : texts) { for (const auto &text : texts) {
do_test(history_contains(everything, text)); do_test(history_contains(everything, text));
} }
@ -4285,6 +4340,86 @@ void history_tests_t::test_history_merge() {
everything->clear(); everything->clear();
} }
void history_tests_t::test_history_path_detection() {
// Regression test for #7582.
say(L"Testing history path detection");
char tmpdirbuff[] = "/tmp/fish_test_history.XXXXXX";
wcstring tmpdir = str2wcstring(mkdtemp(tmpdirbuff));
if (! string_suffixes_string(L"/", tmpdir)) {
tmpdir.push_back(L'/');
}
// Place one valid file in the directory.
wcstring filename = L"testfile";
std::string path = wcs2string(tmpdir + filename);
FILE *f = fopen(path.c_str(), "w");
if (!f) {
err(L"Failed to open test file from history path detection");
return;
}
fclose(f);
std::shared_ptr<test_environment_t> vars = std::make_shared<test_environment_t>();
vars->vars[L"PWD"] = tmpdir;
vars->vars[L"HOME"] = tmpdir;
std::shared_ptr<history_t> history = history_t::with_name(L"path_detection");
history_t::add_pending_with_file_detection(history, L"cmd0 not/a/valid/path", vars);
history_t::add_pending_with_file_detection(history, L"cmd1 " + filename, vars);
history_t::add_pending_with_file_detection(history, L"cmd2 " + tmpdir + L"/" + filename, vars);
history_t::add_pending_with_file_detection(history, L"cmd3 $HOME/" + filename, vars);
history_t::add_pending_with_file_detection(history, L"cmd4 $HOME/notafile", vars);
history_t::add_pending_with_file_detection(history, L"cmd5 ~/" + filename, vars);
history_t::add_pending_with_file_detection(history, L"cmd6 ~/notafile", vars);
history_t::add_pending_with_file_detection(history, L"cmd7 ~/*f*", vars);
history_t::add_pending_with_file_detection(history, L"cmd8 ~/*zzz*", vars);
history->resolve_pending();
constexpr size_t hist_size = 9;
if (history->size() != hist_size) {
err(L"history has wrong size: %lu but expected %lu", (unsigned long)history->size(),
(unsigned long)hist_size);
history->clear();
return;
}
// Expected sets of paths.
wcstring_list_t expected[hist_size] = {
{}, // cmd0
{filename}, // cmd1
{tmpdir + L"/" + filename}, // cmd2
{L"$HOME/" + filename}, // cmd3
{}, // cmd4
{L"~/" + filename}, // cmd5
{}, // cmd6
{}, // cmd7 - we do not expand globs
{}, // cmd8
};
size_t lap;
const size_t maxlap = 128;
for (lap = 0; lap < maxlap; lap++) {
int failures = 0;
bool last = (lap + 1 == maxlap);
for (size_t i = 1; i <= hist_size; i++) {
if (history->item_at_index(i).required_paths != expected[hist_size - i]) {
failures += 1;
if (last) {
err(L"Wrong detected paths for item %lu", (unsigned long)i);
}
}
}
if (failures == 0) {
break;
}
// The file detection takes a little time since it occurs in the background.
// Loop until the test passes.
usleep(1E6 / 500); // 1 msec
}
//fprintf(stderr, "History saving took %lu laps\n", (unsigned long)lap);
history->clear();
}
static bool install_sample_history(const wchar_t *name) { static bool install_sample_history(const wchar_t *name) {
wcstring path; wcstring path;
if (!path_get_data(path)) { if (!path_get_data(path)) {
@ -4301,7 +4436,7 @@ static bool install_sample_history(const wchar_t *name) {
} }
/// Indicates whether the history is equal to the given null-terminated array of strings. /// Indicates whether the history is equal to the given null-terminated array of strings.
static bool history_equals(history_t &hist, const wchar_t *const *strings) { static bool history_equals(const shared_ptr<history_t> &hist, const wchar_t *const *strings) {
// Count our expected items. // Count our expected items.
size_t expected_count = 0; size_t expected_count = 0;
while (strings[expected_count]) { while (strings[expected_count]) {
@ -4313,10 +4448,10 @@ static bool history_equals(history_t &hist, const wchar_t *const *strings) {
size_t array_idx = 0; size_t array_idx = 0;
for (;;) { for (;;) {
const wchar_t *expected = strings[array_idx]; const wchar_t *expected = strings[array_idx];
history_item_t item = hist.item_at_index(history_idx); history_item_t item = hist->item_at_index(history_idx);
if (expected == NULL) { if (expected == NULL) {
if (!item.empty()) { if (!item.empty()) {
err(L"Expected empty item at history index %lu", history_idx); err(L"Expected empty item at history index %lu, instead found: %ls", history_idx, item.str().c_str());
} }
break; break;
} else { } else {
@ -4345,11 +4480,11 @@ void history_tests_t::test_history_formats() {
const wchar_t *const expected[] = { const wchar_t *const expected[] = {
L"#def", L"echo #abc", L"function yay\necho hi\nend", L"cd foobar", L"ls /", NULL}; L"#def", L"echo #abc", L"function yay\necho hi\nend", L"cd foobar", L"ls /", NULL};
history_t &test_history = history_t::history_with_name(name); auto test_history = history_t::with_name(name);
if (!history_equals(test_history, expected)) { if (!history_equals(test_history, expected)) {
err(L"test_history_formats failed for %ls\n", name); err(L"test_history_formats failed for %ls\n", name);
} }
test_history.clear(); test_history->clear();
} }
name = L"history_sample_fish_2_0"; name = L"history_sample_fish_2_0";
@ -4360,11 +4495,11 @@ void history_tests_t::test_history_formats() {
const wchar_t *const expected[] = {L"echo this has\\\nbackslashes", const wchar_t *const expected[] = {L"echo this has\\\nbackslashes",
L"function foo\necho bar\nend", L"echo alpha", NULL}; L"function foo\necho bar\nend", L"echo alpha", NULL};
history_t &test_history = history_t::history_with_name(name); auto test_history = history_t::with_name(name);
if (!history_equals(test_history, expected)) { if (!history_equals(test_history, expected)) {
err(L"test_history_formats failed for %ls\n", name); err(L"test_history_formats failed for %ls\n", name);
} }
test_history.clear(); test_history->clear();
} }
say(L"Testing bash import"); say(L"Testing bash import");
@ -4374,21 +4509,16 @@ void history_tests_t::test_history_formats() {
} else { } else {
// The results are in the reverse order that they appear in the bash history file. // The results are in the reverse order that they appear in the bash history file.
// We don't expect whitespace to be elided (#4908: except for leading/trailing whitespace) // We don't expect whitespace to be elided (#4908: except for leading/trailing whitespace)
const wchar_t *expected[] = {L"/** # see issue 7407", const wchar_t *expected[] = {
L"sleep 123", L"/** # see issue 7407", L"sleep 123", L"a && echo valid construct",
L"a && echo valid construct", L"final line", L"echo supsup", L"export XVAR='exported'",
L"final line", L"history --help", L"echo foo", NULL};
L"echo supsup", auto test_history = history_t::with_name(L"bash_import");
L"export XVAR='exported'", test_history->populate_from_bash(f);
L"history --help",
L"echo foo",
NULL};
history_t &test_history = history_t::history_with_name(L"bash_import");
test_history.populate_from_bash(f);
if (!history_equals(test_history, expected)) { if (!history_equals(test_history, expected)) {
err(L"test_history_formats failed for bash import\n"); err(L"test_history_formats failed for bash import\n");
} }
test_history.clear(); test_history->clear();
fclose(f); fclose(f);
} }
@ -4398,13 +4528,13 @@ void history_tests_t::test_history_formats() {
err(L"Couldn't open file tests/%ls", name); err(L"Couldn't open file tests/%ls", name);
} else { } else {
// We simply invoke get_string_representation. If we don't die, the test is a success. // We simply invoke get_string_representation. If we don't die, the test is a success.
history_t &test_history = history_t::history_with_name(name); auto test_history = history_t::with_name(name);
const wchar_t *expected[] = {L"no_newline_at_end_of_file", L"corrupt_prefix", const wchar_t *expected[] = {L"no_newline_at_end_of_file", L"corrupt_prefix",
L"this_command_is_ok", NULL}; L"this_command_is_ok", NULL};
if (!history_equals(test_history, expected)) { if (!history_equals(test_history, expected)) {
err(L"test_history_formats failed for %ls\n", name); err(L"test_history_formats failed for %ls\n", name);
} }
test_history.clear(); test_history->clear();
} }
} }
@ -4549,6 +4679,10 @@ static bool test_1_parse_ll2(const wcstring &src, wcstring *out_cmd, wcstring *o
statement = tmp; statement = tmp;
} }
} }
if (!statement) {
say(L"No decorated statement found in '%ls'", src.c_str());
return false;
}
// Return its decoration and command. // Return its decoration and command.
*out_deco = statement->decoration(); *out_deco = statement->decoration();
@ -6198,6 +6332,7 @@ int main(int argc, char **argv) {
if (should_test_function("autosuggest_suggest_special")) test_autosuggest_suggest_special(); if (should_test_function("autosuggest_suggest_special")) test_autosuggest_suggest_special();
if (should_test_function("history")) history_tests_t::test_history(); if (should_test_function("history")) history_tests_t::test_history();
if (should_test_function("history_merge")) history_tests_t::test_history_merge(); if (should_test_function("history_merge")) history_tests_t::test_history_merge();
if (should_test_function("history_paths")) history_tests_t::test_history_path_detection();
if (!is_windows_subsystem_for_linux()) { if (!is_windows_subsystem_for_linux()) {
// this test always fails under WSL // this test always fails under WSL
if (should_test_function("history_races")) history_tests_t::test_history_races(); if (should_test_function("history_races")) history_tests_t::test_history_races();

View file

@ -4,6 +4,7 @@
#include <assert.h> #include <assert.h>
#include <atomic>
#include <unordered_map> #include <unordered_map>
#include "common.h" #include "common.h"
@ -28,13 +29,13 @@ class features_t {
/// Return whether a flag is set. /// Return whether a flag is set.
bool test(flag_t f) const { bool test(flag_t f) const {
assert(f >= 0 && f < flag_count && "Invalid flag"); assert(f >= 0 && f < flag_count && "Invalid flag");
return values[f]; return values[f].load(std::memory_order_relaxed);
} }
/// Set a flag. /// Set a flag.
void set(flag_t f, bool value) { void set(flag_t f, bool value) {
assert(f >= 0 && f < flag_count && "Invalid flag"); assert(f >= 0 && f < flag_count && "Invalid flag");
values[f] = value; values[f].store(value, std::memory_order_relaxed);
} }
/// Parses a comma-separated feature-flag string, updating ourselves with the values. /// Parses a comma-separated feature-flag string, updating ourselves with the values.
@ -69,9 +70,20 @@ class features_t {
features_t(); features_t();
features_t(const features_t &rhs) { *this = rhs; }
void operator=(const features_t &rhs) {
for (int i = 0; i < flag_count; i++) {
flag_t f = static_cast<flag_t>(i);
this->set(f, rhs.test(f));
}
}
private: private:
/// Values for the flags. // Values for the flags.
bool values[flag_count] = {}; // These are atomic to "fix" a race reported by tsan where tests of feature flags and other
// tests which use them conceptually race.
std::atomic<bool> values[flag_count]{};
}; };
/// Return the global set of features for fish. This is const to prevent accidental mutation. /// Return the global set of features for fish. This is const to prevent accidental mutation.

View file

@ -468,7 +468,7 @@ bool autosuggest_validate_from_history(const history_item_t &item,
} }
// Did the historical command have arguments that look like paths, which aren't paths now? // Did the historical command have arguments that look like paths, which aren't paths now?
if (!all_paths_are_valid(item.get_required_paths(), working_directory)) { if (!all_paths_are_valid(item.get_required_paths(), ctx)) {
return false; return false;
} }

View file

@ -172,22 +172,28 @@ class history_lru_cache_t : public lru_cache_t<history_lru_cache_t, history_item
/// We can merge two items if they are the same command. We use the more recent timestamp, more /// We can merge two items if they are the same command. We use the more recent timestamp, more
/// recent identifier, and the longer list of required paths. /// recent identifier, and the longer list of required paths.
bool history_item_t::merge(const history_item_t &item) { bool history_item_t::merge(const history_item_t &item) {
bool result = false; // We can only merge items if they agree on their text and persistence mode.
if (this->contents == item.contents) { if (this->contents != item.contents || this->persist_mode != item.persist_mode) {
this->creation_timestamp = std::max(this->creation_timestamp, item.creation_timestamp); return false;
if (this->required_paths.size() < item.required_paths.size()) {
this->required_paths = item.required_paths;
}
if (this->identifier < item.identifier) {
this->identifier = item.identifier;
}
result = true;
} }
return result;
// Ok, merge this item.
this->creation_timestamp = std::max(this->creation_timestamp, item.creation_timestamp);
if (this->required_paths.size() < item.required_paths.size()) {
this->required_paths = item.required_paths;
}
if (this->identifier < item.identifier) {
this->identifier = item.identifier;
}
return true;
} }
history_item_t::history_item_t(wcstring str, time_t when, history_identifier_t ident) history_item_t::history_item_t(wcstring str, time_t when, history_identifier_t ident,
: contents(trim(std::move(str))), creation_timestamp(when), identifier(ident) {} history_persistence_mode_t persist_mode)
: contents(std::move(str)),
creation_timestamp(when),
identifier(ident),
persist_mode(persist_mode) {}
bool history_item_t::matches_search(const wcstring &term, enum history_search_type_t type, bool history_item_t::matches_search(const wcstring &term, enum history_search_type_t type,
bool case_sensitive) const { bool case_sensitive) const {
@ -228,9 +234,11 @@ bool history_item_t::matches_search(const wcstring &term, enum history_search_ty
} }
struct history_impl_t { struct history_impl_t {
// Privately add an item. If pending, the item will not be returned by history searches until a // Add a new history item to the end. If pending is set, the item will not be returned by
// call to resolve_pending. // item_at_index until a call to resolve_pending(). Pending items are tracked with an offset
void add(const history_item_t &item, bool pending = false, bool do_save = true); // into the array of new items, so adding a non-pending item has the effect of resolving all
// pending items.
void add(history_item_t item, bool pending = false, bool do_save = true);
// Internal function. // Internal function.
void clear_file_state(); void clear_file_state();
@ -266,7 +274,10 @@ struct history_impl_t {
// the boundary are considered "old". Items whose timestemps are > the boundary are new, and are // the boundary are considered "old". Items whose timestemps are > the boundary are new, and are
// ignored by this instance (unless they came from this instance). The timestamp may be adjusted // ignored by this instance (unless they came from this instance). The timestamp may be adjusted
// by incorporate_external_changes(). // by incorporate_external_changes().
time_t boundary_timestamp{time(nullptr)}; time_t boundary_timestamp{};
/// The most recent "unique" identifier for a history item.
history_identifier_t last_identifier{0};
// How many items we add until the next vacuum. Initially a random value. // How many items we add until the next vacuum. Initially a random value.
int countdown_to_vacuum{-1}; int countdown_to_vacuum{-1};
@ -277,6 +288,12 @@ struct history_impl_t {
// List of old items, as offsets into out mmap data. // List of old items, as offsets into out mmap data.
std::deque<size_t> old_item_offsets{}; std::deque<size_t> old_item_offsets{};
/// \return a timestamp for new items - see the implementation for a subtlety.
time_t timestamp_now() const;
/// \return a new item identifier, incrementing our counter.
history_identifier_t next_identifier() { return ++last_identifier; }
// Figure out the offsets of our file contents. // Figure out the offsets of our file contents.
void populate_from_file_contents(); void populate_from_file_contents();
@ -286,6 +303,11 @@ struct history_impl_t {
// Deletes duplicates in new_items. // Deletes duplicates in new_items.
void compact_new_items(); void compact_new_items();
// Removes trailing ephemeral items.
// Ephemeral items have leading spaces, and can only be retrieved immediately; adding any item
// removes them.
void remove_ephemeral_items();
// Attempts to rewrite the existing file to a target temporary file // Attempts to rewrite the existing file to a target temporary file
// Returns false on error, true on success // Returns false on error, true on success
bool rewrite_to_temporary_file(int existing_fd, int dst_fd) const; bool rewrite_to_temporary_file(int existing_fd, int dst_fd) const;
@ -302,7 +324,9 @@ struct history_impl_t {
// Saves history unless doing so is disabled. // Saves history unless doing so is disabled.
void save_unless_disabled(); void save_unless_disabled();
explicit history_impl_t(wcstring name) : name(std::move(name)) {} explicit history_impl_t(wcstring name)
: name(std::move(name)), boundary_timestamp(time(nullptr)) {}
history_impl_t(history_impl_t &&) = default; history_impl_t(history_impl_t &&) = default;
~history_impl_t() = default; ~history_impl_t() = default;
@ -313,20 +337,9 @@ struct history_impl_t {
// require populating the history. // require populating the history.
bool is_empty(); bool is_empty();
// Add a new history item to the end. If pending is set, the item will not be returned by
// item_at_index until a call to resolve_pending(). Pending items are tracked with an offset
// into the array of new items, so adding a non-pending item has the effect of resolving all
// pending items.
void add(const wcstring &str, history_identifier_t ident = 0, bool pending = false,
bool save = true);
// Remove a history item. // Remove a history item.
void remove(const wcstring &str); void remove(const wcstring &str);
// Add a new pending history item to the end, and then begin file detection on the items to
// determine which arguments are paths
void add_pending_with_file_detection(const wcstring &str, const wcstring &working_dir_slash);
// Resolves any pending history items, so that they may be returned in history searches. // Resolves any pending history items, so that they may be returned in history searches.
void resolve_pending(); void resolve_pending();
@ -356,7 +369,7 @@ struct history_impl_t {
std::unordered_map<long, wcstring> items_at_indexes(const std::vector<long> &idxs); std::unordered_map<long, wcstring> items_at_indexes(const std::vector<long> &idxs);
// Sets the valid file paths for the history item with the given identifier. // Sets the valid file paths for the history item with the given identifier.
void set_valid_file_paths(const wcstring_list_t &valid_file_paths, history_identifier_t ident); void set_valid_file_paths(wcstring_list_t &&valid_file_paths, history_identifier_t ident);
// Return the specified history at the specified index. 0 is the index of the current // Return the specified history at the specified index. 0 is the index of the current
// commandline. (So the most recent item is at index 1.) // commandline. (So the most recent item is at index 1.)
@ -366,12 +379,14 @@ struct history_impl_t {
size_t size(); size_t size();
}; };
void history_impl_t::add(const history_item_t &item, bool pending, bool do_save) { void history_impl_t::add(history_item_t item, bool pending, bool do_save) {
assert(item.timestamp() != 0 && "Should not add an item with a 0 timestamp");
// We use empty items as sentinels to indicate the end of history. // We use empty items as sentinels to indicate the end of history.
// Do not allow them to be added (#6032). // Do not allow them to be added (#6032).
if (item.contents.empty()) { if (item.contents.empty()) {
return; return;
} }
// Try merging with the last item. // Try merging with the last item.
if (!new_items.empty() && new_items.back().merge(item)) { if (!new_items.empty() && new_items.back().merge(item)) {
// We merged, so we don't have to add anything. Maybe this item was pending, but it just got // We merged, so we don't have to add anything. Maybe this item was pending, but it just got
@ -422,20 +437,6 @@ void history_impl_t::save_unless_disabled() {
countdown_to_vacuum--; countdown_to_vacuum--;
} }
void history_impl_t::add(const wcstring &str, history_identifier_t ident, bool pending,
bool do_save) {
time_t when = time(nullptr);
// Big hack: do not allow timestamps equal to our boundary date. This is because we include
// items whose timestamps are equal to our boundary when reading old history, so we can catch
// "just closed" items. But this means that we may interpret our own items, that we just wrote,
// as old items, if we wrote them in the same second as our birthdate.
if (when == this->boundary_timestamp) {
when++;
}
this->add(history_item_t(str, when, ident), pending, do_save);
}
// Remove matching history entries from our list of new items. This only supports literal, // Remove matching history entries from our list of new items. This only supports literal,
// case-sensitive, matches. // case-sensitive, matches.
void history_impl_t::remove(const wcstring &str_to_remove) { void history_impl_t::remove(const wcstring &str_to_remove) {
@ -458,7 +459,7 @@ void history_impl_t::remove(const wcstring &str_to_remove) {
assert(first_unwritten_new_item_index <= new_items.size()); assert(first_unwritten_new_item_index <= new_items.size());
} }
void history_impl_t::set_valid_file_paths(const wcstring_list_t &valid_file_paths, void history_impl_t::set_valid_file_paths(wcstring_list_t &&valid_file_paths,
history_identifier_t ident) { history_identifier_t ident) {
// 0 identifier is used to mean "not necessary". // 0 identifier is used to mean "not necessary".
if (ident == 0) { if (ident == 0) {
@ -468,7 +469,7 @@ void history_impl_t::set_valid_file_paths(const wcstring_list_t &valid_file_path
// Look for an item with the given identifier. It is likely to be at the end of new_items. // Look for an item with the given identifier. It is likely to be at the end of new_items.
for (auto iter = new_items.rbegin(); iter != new_items.rend(); ++iter) { for (auto iter = new_items.rbegin(); iter != new_items.rend(); ++iter) {
if (iter->identifier == ident) { // found it if (iter->identifier == ident) { // found it
iter->required_paths = valid_file_paths; iter->required_paths = std::move(valid_file_paths);
break; break;
} }
} }
@ -557,6 +558,18 @@ std::unordered_map<long, wcstring> history_impl_t::items_at_indexes(const std::v
return result; return result;
} }
time_t history_impl_t::timestamp_now() const {
time_t when = time(nullptr);
// Big hack: do not allow timestamps equal to our boundary date. This is because we include
// items whose timestamps are equal to our boundary when reading old history, so we can catch
// "just closed" items. But this means that we may interpret our own items, that we just wrote,
// as old items, if we wrote them in the same second as our birthdate.
if (when == this->boundary_timestamp) {
when++;
}
return when;
}
void history_impl_t::populate_from_file_contents() { void history_impl_t::populate_from_file_contents() {
old_item_offsets.clear(); old_item_offsets.clear();
if (file_contents) { if (file_contents) {
@ -648,12 +661,15 @@ void history_impl_t::clear_file_state() {
} }
void history_impl_t::compact_new_items() { void history_impl_t::compact_new_items() {
// Keep only the most recent items with the given contents. This algorithm could be made more // Keep only the most recent items with the given contents.
// efficient, but likely would consume more memory too.
std::unordered_set<wcstring> seen; std::unordered_set<wcstring> seen;
size_t idx = new_items.size(); size_t idx = new_items.size();
while (idx--) { while (idx--) {
const history_item_t &item = new_items[idx]; const history_item_t &item = new_items[idx];
// Only compact persisted items.
if (!item.should_write_to_disk()) continue;
if (!seen.insert(item.contents).second) { if (!seen.insert(item.contents).second) {
// This item was not inserted because it was already in the set, so delete the item at // This item was not inserted because it was already in the set, so delete the item at
// this index. // this index.
@ -668,6 +684,14 @@ void history_impl_t::compact_new_items() {
} }
} }
void history_impl_t::remove_ephemeral_items() {
while (!new_items.empty() &&
new_items.back().persist_mode == history_persistence_mode_t::ephemeral) {
new_items.pop_back();
}
first_unwritten_new_item_index = std::min(first_unwritten_new_item_index, new_items.size());
}
// Given the fd of an existing history file, or -1 if none, write // Given the fd of an existing history file, or -1 if none, write
// a new history file to temp_fd. Returns true on success, false // a new history file to temp_fd. Returns true on success, false
// on error // on error
@ -698,7 +722,9 @@ bool history_impl_t::rewrite_to_temporary_file(int existing_fd, int dst_fd) cons
// Insert any unwritten new items // Insert any unwritten new items
for (auto iter = new_items.cbegin() + this->first_unwritten_new_item_index; for (auto iter = new_items.cbegin() + this->first_unwritten_new_item_index;
iter != new_items.cend(); ++iter) { iter != new_items.cend(); ++iter) {
lru.add_item(*iter); if (iter->should_write_to_disk()) {
lru.add_item(*iter);
}
} }
// Stable-sort our items by timestamp // Stable-sort our items by timestamp
@ -920,10 +946,12 @@ bool history_impl_t::save_internal_via_appending() {
std::string buffer; std::string buffer;
while (first_unwritten_new_item_index < new_items.size()) { while (first_unwritten_new_item_index < new_items.size()) {
const history_item_t &item = new_items.at(first_unwritten_new_item_index); const history_item_t &item = new_items.at(first_unwritten_new_item_index);
append_history_item_to_buffer(item, &buffer); if (item.should_write_to_disk()) {
err = flush_to_fd(&buffer, history_fd.fd(), HISTORY_OUTPUT_BUFFER_SIZE); append_history_item_to_buffer(item, &buffer);
if (err) break; err = flush_to_fd(&buffer, history_fd.fd(), HISTORY_OUTPUT_BUFFER_SIZE);
// We wrote this item, hooray. if (err) break;
}
// We wrote or skipped this item, hooray.
first_unwritten_new_item_index++; first_unwritten_new_item_index++;
} }
@ -1057,6 +1085,12 @@ bool history_impl_t::is_empty() {
return empty; return empty;
} }
void history_t::add(wcstring str) {
auto imp = this->impl();
time_t when = imp->timestamp_now();
imp->add(history_item_t(std::move(str), when));
}
/// Populates from older location (in config path, rather than data path) This is accomplished by /// Populates from older location (in config path, rather than data path) This is accomplished by
/// clearing ourselves, and copying the contents of the old history file to the new history file. /// clearing ourselves, and copying the contents of the old history file to the new history file.
/// The new contents will automatically be re-mapped later. /// The new contents will automatically be re-mapped later.
@ -1132,6 +1166,8 @@ static bool should_import_bash_history_line(const wcstring &line) {
/// encode multiline commands. /// encode multiline commands.
void history_impl_t::populate_from_bash(FILE *stream) { void history_impl_t::populate_from_bash(FILE *stream) {
// Process the entire history file until EOF is observed. // Process the entire history file until EOF is observed.
// Pretend all items were created at this time.
const auto when = this->timestamp_now();
bool eof = false; bool eof = false;
while (!eof) { while (!eof) {
auto line = std::string(); auto line = std::string();
@ -1151,10 +1187,11 @@ void history_impl_t::populate_from_bash(FILE *stream) {
if (a_newline) break; if (a_newline) break;
} }
wcstring wide_line = str2wcstring(line); wcstring wide_line = trim(str2wcstring(line));
// Add this line if it doesn't contain anything we know we can't handle. // Add this line if it doesn't contain anything we know we can't handle.
if (should_import_bash_history_line(wide_line)) { if (should_import_bash_history_line(wide_line)) {
this->add(wide_line, 0, false /* pending */, false /* do_save */); this->add(history_item_t(std::move(wide_line), when), false /* pending */,
false /* do_save */);
} }
} }
this->save_unless_disabled(); this->save_unless_disabled();
@ -1163,7 +1200,7 @@ void history_impl_t::populate_from_bash(FILE *stream) {
void history_impl_t::incorporate_external_changes() { void history_impl_t::incorporate_external_changes() {
// To incorporate new items, we simply update our timestamp to now, so that items from previous // To incorporate new items, we simply update our timestamp to now, so that items from previous
// instances get added. We then clear the file state so that we remap the file. Note that this // instances get added. We then clear the file state so that we remap the file. Note that this
// is somehwhat expensive because we will be going back over old items. An optimization would be // is somewhat expensive because we will be going back over old items. An optimization would be
// to preserve old_item_offsets so that they don't have to be recomputed. (However, then items // to preserve old_item_offsets so that they don't have to be recomputed. (However, then items
// *deleted* in other instances would not show up here). // *deleted* in other instances would not show up here).
time_t new_timestamp = time(nullptr); time_t new_timestamp = time(nullptr);
@ -1174,9 +1211,11 @@ void history_impl_t::incorporate_external_changes() {
this->boundary_timestamp = new_timestamp; this->boundary_timestamp = new_timestamp;
this->clear_file_state(); this->clear_file_state();
// We also need to erase new_items, since we go through those first, and that means we // We also need to erase new items, since we go through those first, and that means we
// will not properly interleave them with items from other instances. // will not properly interleave them with items from other instances.
// We'll pick them up from the file (#2312) // We'll pick them up from the file (#2312)
// TODO: this will drop items that had no_persist set, how can we avoid that while still
// properly interleaving?
this->save(false); this->save(false);
this->new_items.clear(); this->new_items.clear();
this->first_unwritten_new_item_index = 0; this->first_unwritten_new_item_index = 0;
@ -1207,21 +1246,42 @@ wcstring history_session_id(const environment_t &vars) {
return result; return result;
} }
path_list_t valid_paths(const path_list_t &paths, const wcstring &working_directory) { path_list_t expand_and_detect_paths(const path_list_t &paths, const environment_t &vars) {
ASSERT_IS_BACKGROUND_THREAD(); ASSERT_IS_BACKGROUND_THREAD();
wcstring_list_t result; wcstring_list_t result;
wcstring working_directory = vars.get_pwd_slash();
operation_context_t ctx(vars, kExpansionLimitBackground);
for (const wcstring &path : paths) { for (const wcstring &path : paths) {
if (path_is_valid(path, working_directory)) { // Suppress cmdsubs since we are on a background thread and don't want to execute fish
result.push_back(path); // script.
// Suppress wildcards because we want to suggest e.g. `rm *` even if the directory
// is empty (and so rm will fail); this is nevertheless a useful command because it
// confirms the directory is empty.
wcstring expanded_path = path;
if (expand_one(expanded_path, {expand_flag::skip_cmdsubst, expand_flag::skip_wildcards},
ctx)) {
if (path_is_valid(expanded_path, working_directory)) {
// Note we return the original (unexpanded) path.
result.push_back(path);
}
} }
} }
return result; return result;
} }
bool all_paths_are_valid(const path_list_t &paths, const wcstring &working_directory) { bool all_paths_are_valid(const path_list_t &paths, const operation_context_t &ctx) {
ASSERT_IS_BACKGROUND_THREAD(); ASSERT_IS_BACKGROUND_THREAD();
wcstring working_directory = ctx.vars.get_pwd_slash();
for (const wcstring &path : paths) { for (const wcstring &path : paths) {
if (!path_is_valid(path, working_directory)) { if (ctx.cancel_checker()) {
return false;
}
wcstring expanded_path = path;
if (!expand_one(expanded_path, {expand_flag::skip_cmdsubst, expand_flag::skip_wildcards},
ctx)) {
return false;
}
if (!path_is_valid(expanded_path, working_directory)) {
return false; return false;
} }
} }
@ -1258,16 +1318,19 @@ bool history_t::is_default() const { return impl()->is_default(); }
bool history_t::is_empty() { return impl()->is_empty(); } bool history_t::is_empty() { return impl()->is_empty(); }
void history_t::add(const history_item_t &item, bool pending) { impl()->add(item, pending); } void history_t::add(history_item_t item, bool pending) { impl()->add(std::move(item), pending); }
void history_t::add(const wcstring &str, history_identifier_t ident, bool pending) {
impl()->add(str, ident, pending);
}
void history_t::remove(const wcstring &str) { impl()->remove(str); } void history_t::remove(const wcstring &str) { impl()->remove(str); }
void history_t::add_pending_with_file_detection(const wcstring &str, void history_t::remove_ephemeral_items() { impl()->remove_ephemeral_items(); }
const wcstring &working_dir_slash) {
// static
void history_t::add_pending_with_file_detection(const std::shared_ptr<history_t> &self,
const wcstring &str,
const std::shared_ptr<environment_t> &vars,
history_persistence_mode_t persist_mode) {
assert(self && "Null history");
// We use empty items as sentinels to indicate the end of history. // We use empty items as sentinels to indicate the end of history.
// Do not allow them to be added (#6032). // Do not allow them to be added (#6032).
if (str.empty()) { if (str.empty()) {
@ -1283,9 +1346,8 @@ void history_t::add_pending_with_file_detection(const wcstring &str,
for (const node_t &node : ast) { for (const node_t &node : ast) {
if (const argument_t *arg = node.try_as<argument_t>()) { if (const argument_t *arg = node.try_as<argument_t>()) {
wcstring potential_path = arg->source(str); wcstring potential_path = arg->source(str);
bool unescaped = unescape_string_in_place(&potential_path, UNESCAPE_DEFAULT); if (string_could_be_path(potential_path)) {
if (unescaped && string_could_be_path(potential_path)) { potential_paths.push_back(std::move(potential_path));
potential_paths.push_back(potential_path);
} }
} else if (const decorated_statement_t *stmt = node.try_as<decorated_statement_t>()) { } else if (const decorated_statement_t *stmt = node.try_as<decorated_statement_t>()) {
// Hack hack hack - if the command is likely to trigger an exit, then don't do // Hack hack hack - if the command is likely to trigger an exit, then don't do
@ -1309,30 +1371,32 @@ void history_t::add_pending_with_file_detection(const wcstring &str,
// If we got a path, we'll perform file detection for autosuggestion hinting. // If we got a path, we'll perform file detection for autosuggestion hinting.
bool wants_file_detection = !potential_paths.empty() && !needs_sync_write; bool wants_file_detection = !potential_paths.empty() && !needs_sync_write;
auto imp = this->impl(); auto imp = self->impl();
// Make our history item.
time_t when = imp->timestamp_now();
history_identifier_t identifier = imp->next_identifier();
history_item_t item{str, when, identifier, persist_mode};
history_identifier_t identifier = 0;
if (wants_file_detection) { if (wants_file_detection) {
// Grab the next identifier.
static relaxed_atomic_t<history_identifier_t> s_last_identifier{0};
identifier = ++s_last_identifier;
imp->disable_automatic_saving(); imp->disable_automatic_saving();
// Add the item. Then check for which paths are valid on a background thread, // Add the item. Then check for which paths are valid on a background thread,
// and unblock the item. // and unblock the item.
// Don't hold the lock while we perform this file detection. // Don't hold the lock while we perform this file detection.
imp->add(str, identifier, true /* pending */); imp->add(std::move(item), true /* pending */);
iothread_perform([=]() { iothread_perform([=]() {
auto validated_paths = valid_paths(potential_paths, working_dir_slash); // Don't hold the lock while we perform this file detection.
auto imp = this->impl(); auto validated_paths = expand_and_detect_paths(potential_paths, *vars);
imp->set_valid_file_paths(validated_paths, identifier); auto imp = self->impl();
imp->set_valid_file_paths(std::move(validated_paths), identifier);
imp->enable_automatic_saving(); imp->enable_automatic_saving();
}); });
} else { } else {
// Add the item. // Add the item.
// If we think we're about to exit, save immediately, regardless of any disabling. This may // If we think we're about to exit, save immediately, regardless of any disabling. This may
// cause us to lose file hinting for some commands, but it beats losing history items. // cause us to lose file hinting for some commands, but it beats losing history items.
imp->add(str, identifier, true /* pending */); imp->add(std::move(item), true /* pending */);
if (needs_sync_write) { if (needs_sync_write) {
imp->save(); imp->save();
} }
@ -1344,7 +1408,7 @@ void history_t::save() { impl()->save(); }
/// Perform a search of \p hist for \p search_string. Invoke a function \p func for each match. If /// Perform a search of \p hist for \p search_string. Invoke a function \p func for each match. If
/// \p func returns true, continue the search; else stop it. /// \p func returns true, continue the search; else stop it.
static void do_1_history_search(history_t &hist, history_search_type_t search_type, static void do_1_history_search(history_t *hist, history_search_type_t search_type,
const wcstring &search_string, bool case_sensitive, const wcstring &search_string, bool case_sensitive,
const std::function<bool(const history_item_t &item)> &func, const std::function<bool(const history_item_t &item)> &func,
const cancel_checker_t &cancel_check) { const cancel_checker_t &cancel_check) {
@ -1383,7 +1447,7 @@ bool history_t::search(history_search_type_t search_type, const wcstring_list_t
if (search_args.empty()) { if (search_args.empty()) {
// The user had no search terms; just append everything. // The user had no search terms; just append everything.
do_1_history_search(*this, history_search_type_t::match_everything, {}, false, func, do_1_history_search(this, history_search_type_t::match_everything, {}, false, func,
cancel_check); cancel_check);
} else { } else {
for (const wcstring &search_string : search_args) { for (const wcstring &search_string : search_args) {
@ -1391,7 +1455,7 @@ bool history_t::search(history_search_type_t search_type, const wcstring_list_t
streams.err.append_format(L"Searching for the empty string isn't allowed"); streams.err.append_format(L"Searching for the empty string isn't allowed");
return false; return false;
} }
do_1_history_search(*this, search_type, search_string, case_sensitive, func, do_1_history_search(this, search_type, search_string, case_sensitive, func,
cancel_check); cancel_check);
} }
} }
@ -1422,7 +1486,7 @@ history_item_t history_t::item_at_index(size_t idx) { return impl()->item_at_ind
size_t history_t::size() { return impl()->size(); } size_t history_t::size() { return impl()->size(); }
/// The set of all histories. /// The set of all histories.
static owning_lock<std::map<wcstring, std::unique_ptr<history_t>>> s_histories; static owning_lock<std::map<wcstring, std::shared_ptr<history_t>>> s_histories;
void history_save_all() { void history_save_all() {
auto histories = s_histories.acquire(); auto histories = s_histories.acquire();
@ -1431,24 +1495,20 @@ void history_save_all() {
} }
} }
history_t &history_t::history_with_name(const wcstring &name) { std::shared_ptr<history_t> history_t::with_name(const wcstring &name) {
// Return a history for the given name, creating it if necessary
// Note that histories are currently never deleted, so we can return a reference to them without
// using something like shared_ptr
auto hs = s_histories.acquire(); auto hs = s_histories.acquire();
std::unique_ptr<history_t> &hist = (*hs)[name]; std::shared_ptr<history_t> &hist = (*hs)[name];
if (!hist) { if (!hist) {
hist = make_unique<history_t>(name); hist = std::make_shared<history_t>(name);
} }
return *hist; return hist;
} }
static relaxed_atomic_bool_t private_mode{false};
void start_private_mode(env_stack_t &vars) { void start_private_mode(env_stack_t &vars) {
private_mode = true;
vars.set_one(L"fish_history", ENV_GLOBAL, L""); vars.set_one(L"fish_history", ENV_GLOBAL, L"");
vars.set_one(L"fish_private_mode", ENV_GLOBAL, L"1"); vars.set_one(L"fish_private_mode", ENV_GLOBAL, L"1");
} }
bool in_private_mode() { return private_mode; } bool in_private_mode(const environment_t &vars) {
return !vars.get(L"fish_private_mode").missing_or_empty();
}

View file

@ -24,6 +24,7 @@
struct io_streams_t; struct io_streams_t;
class env_stack_t; class env_stack_t;
class environment_t; class environment_t;
class operation_context_t;
// Fish supports multiple shells writing to history at once. Here is its strategy: // Fish supports multiple shells writing to history at once. Here is its strategy:
// //
@ -62,11 +63,41 @@ enum class history_search_type_t {
typedef uint64_t history_identifier_t; typedef uint64_t history_identifier_t;
/// Ways that a history item may be written to disk (or omitted).
enum class history_persistence_mode_t : uint8_t {
disk, // the history item is written to disk normally
memory, // the history item is stored in-memory only, not written to disk
ephemeral, // the history item is stored in-memory and deleted when a new item is added
};
class history_item_t { class history_item_t {
friend class history_t; public:
friend struct history_impl_t; /// Construct from a text, timestamp, and optional identifier.
friend class history_lru_cache_t; /// If \p no_persist is set, then do not write this item to disk.
friend class history_tests_t; explicit history_item_t(
wcstring str = {}, time_t when = 0, history_identifier_t ident = 0,
history_persistence_mode_t persist_mode = history_persistence_mode_t::disk);
/// \return the text as a string.
const wcstring &str() const { return contents; }
/// \return whether the text is empty.
bool empty() const { return contents.empty(); }
// \return wehther our contents matches a search term.
bool matches_search(const wcstring &term, enum history_search_type_t type,
bool case_sensitive) const;
/// \return the timestamp for creating this history item.
time_t timestamp() const { return creation_timestamp; }
/// \return whether this item should be persisted (written to disk).
bool should_write_to_disk() const { return persist_mode == history_persistence_mode_t::disk; }
/// Get and set the list of arguments which referred to files.
/// This is used for autosuggestion hinting.
const path_list_t &get_required_paths() const { return required_paths; }
void set_required_paths(path_list_t paths) { required_paths = std::move(paths); }
private: private:
// Attempts to merge two compatible history items together. // Attempts to merge two compatible history items together.
@ -78,33 +109,19 @@ class history_item_t {
// Original creation time for the entry. // Original creation time for the entry.
time_t creation_timestamp; time_t creation_timestamp;
// Sometimes unique identifier used for hinting.
history_identifier_t identifier;
// Paths that we require to be valid for this item to be autosuggested. // Paths that we require to be valid for this item to be autosuggested.
path_list_t required_paths; path_list_t required_paths;
public: // Sometimes unique identifier used for hinting.
explicit history_item_t(wcstring str = wcstring(), time_t when = 0, history_identifier_t identifier;
history_identifier_t ident = 0);
const wcstring &str() const { return contents; } // If set, do not write this item to disk.
history_persistence_mode_t persist_mode;
bool empty() const { return contents.empty(); } friend class history_t;
friend struct history_impl_t;
// Whether our contents matches a search term. friend class history_lru_cache_t;
bool matches_search(const wcstring &term, enum history_search_type_t type, friend class history_tests_t;
bool case_sensitive) const;
time_t timestamp() const { return creation_timestamp; }
const path_list_t &get_required_paths() const { return required_paths; }
void set_required_paths(const path_list_t &paths) { required_paths = paths; }
bool operator==(const history_item_t &other) const {
return contents == other.contents && creation_timestamp == other.creation_timestamp &&
required_paths == other.required_paths;
}
}; };
typedef std::deque<history_item_t> history_item_list_t; typedef std::deque<history_item_t> history_item_list_t;
@ -128,8 +145,11 @@ class history_t {
acquired_lock<const history_impl_t> impl() const; acquired_lock<const history_impl_t> impl() const;
// Privately add an item. If pending, the item will not be returned by history searches until a // Privately add an item. If pending, the item will not be returned by history searches until a
// call to resolve_pending. // call to resolve_pending. Any trailing ephemeral items are dropped.
void add(const history_item_t &item, bool pending = false); void add(history_item_t item, bool pending = false);
// Add a new history item with text \p str to the end of history.
void add(wcstring str);
public: public:
explicit history_t(wcstring name); explicit history_t(wcstring name);
@ -143,7 +163,7 @@ class history_t {
static bool never_mmap; static bool never_mmap;
// Returns history with the given name, creating it if necessary. // Returns history with the given name, creating it if necessary.
static history_t &history_with_name(const wcstring &name); static std::shared_ptr<history_t> with_name(const wcstring &name);
/// Returns whether this is using the default name. /// Returns whether this is using the default name.
bool is_default() const; bool is_default() const;
@ -152,18 +172,19 @@ class history_t {
// require populating the history. // require populating the history.
bool is_empty(); bool is_empty();
// Add a new history item to the end. If pending is set, the item will not be returned by
// item_at_index until a call to resolve_pending(). Pending items are tracked with an offset
// into the array of new items, so adding a non-pending item has the effect of resolving all
// pending items.
void add(const wcstring &str, history_identifier_t ident = 0, bool pending = false);
// Remove a history item. // Remove a history item.
void remove(const wcstring &str); void remove(const wcstring &str);
/// Remove any trailing ephemeral items.
void remove_ephemeral_items();
// Add a new pending history item to the end, and then begin file detection on the items to // Add a new pending history item to the end, and then begin file detection on the items to
// determine which arguments are paths // determine which arguments are paths. Arguments may be expanded (e.g. with PWD and variables)
void add_pending_with_file_detection(const wcstring &str, const wcstring &working_dir_slash); // using the given \p vars. The item has the given \p persist_mode.
static void add_pending_with_file_detection(
const std::shared_ptr<history_t> &self, const wcstring &str,
const std::shared_ptr<environment_t> &vars,
history_persistence_mode_t persist_mode = history_persistence_mode_t::disk);
// Resolves any pending history items, so that they may be returned in history searches. // Resolves any pending history items, so that they may be returned in history searches.
void resolve_pending(); void resolve_pending();
@ -221,6 +242,7 @@ using history_search_flags_t = uint32_t;
class history_search_t { class history_search_t {
private: private:
// The history in which we are searching. // The history in which we are searching.
// TODO: this should be a shared_ptr.
history_t *history_; history_t *history_;
// The original search term. // The original search term.
@ -263,16 +285,23 @@ class history_search_t {
// Returns the current search result item contents. asserts if there is no current item. // Returns the current search result item contents. asserts if there is no current item.
const wcstring &current_string() const; const wcstring &current_string() const;
// Constructor. // Construct from a history pointer; the caller is responsible for ensuring the history stays
history_search_t(history_t &hist, const wcstring &str, // alive.
history_search_t(history_t *hist, const wcstring &str,
enum history_search_type_t type = history_search_type_t::contains, enum history_search_type_t type = history_search_type_t::contains,
history_search_flags_t flags = 0) history_search_flags_t flags = 0)
: history_(&hist), orig_term_(str), canon_term_(str), search_type_(type), flags_(flags) { : history_(hist), orig_term_(str), canon_term_(str), search_type_(type), flags_(flags) {
if (ignores_case()) { if (ignores_case()) {
std::transform(canon_term_.begin(), canon_term_.end(), canon_term_.begin(), towlower); std::transform(canon_term_.begin(), canon_term_.end(), canon_term_.begin(), towlower);
} }
} }
// Construct from a shared_ptr. TODO: this should be the only constructor.
history_search_t(const std::shared_ptr<history_t> &hist, const wcstring &str,
enum history_search_type_t type = history_search_type_t::contains,
history_search_flags_t flags = 0)
: history_search_t(hist.get(), str, type, flags) {}
// Default constructor. // Default constructor.
history_search_t() = default; history_search_t() = default;
}; };
@ -283,18 +312,23 @@ void history_save_all();
/// Return the prefix for the files to be used for command and read history. /// Return the prefix for the files to be used for command and read history.
wcstring history_session_id(const environment_t &vars); wcstring history_session_id(const environment_t &vars);
/// Given a list of paths and a working directory, return the paths that are valid /// Given a list of proposed paths and a context, perform variable and home directory expansion,
/// This does disk I/O and may only be called in a background thread /// and detect if the result expands to a value which is also the path to a file.
path_list_t valid_paths(const path_list_t &paths, const wcstring &working_directory); /// Wildcard expansions are suppressed - see implementation comments for why.
/// This is used for autosuggestion hinting. If we add an item to history, and one of its arguments
/// refers to a file, then we only want to suggest it if there is a valid file there.
/// This does disk I/O and may only be called in a background thread.
path_list_t expand_and_detect_paths(const path_list_t &paths, const environment_t &vars);
/// Given a list of paths and a working directory, /// Given a list of proposed paths and a context, expand each one and see if it refers to a file.
/// return true if all paths in the list are valid /// Wildcard expansions are suppressed.
/// Returns true for if paths is empty /// \return true if \p paths is empty or every path is valid.
bool all_paths_are_valid(const path_list_t &paths, const wcstring &working_directory); bool all_paths_are_valid(const path_list_t &paths, const operation_context_t &ctx);
/// Sets private mode on. Once in private mode, it cannot be turned off. /// Sets private mode on. Once in private mode, it cannot be turned off.
void start_private_mode(env_stack_t &vars); void start_private_mode(env_stack_t &vars);
/// Queries private mode status. /// Queries private mode status.
bool in_private_mode(); bool in_private_mode(const environment_t &vars);
#endif #endif

View file

@ -280,7 +280,7 @@ static history_item_t decode_item_fish_2_0(const char *base, size_t len) {
done: done:
history_item_t result(cmd, when); history_item_t result(cmd, when);
result.set_required_paths(paths); result.set_required_paths(std::move(paths));
return result; return result;
} }
@ -431,6 +431,7 @@ static size_t offset_of_next_item_fish_2_0(const history_file_contents_t &conten
} }
void append_history_item_to_buffer(const history_item_t &item, std::string *buffer) { void append_history_item_to_buffer(const history_item_t &item, std::string *buffer) {
assert(item.should_write_to_disk() && "Item should not be persisted");
auto append = [=](const char *a, const char *b = nullptr, const char *c = nullptr) { auto append = [=](const char *a, const char *b = nullptr, const char *c = nullptr) {
if (a) buffer->append(a); if (a) buffer->append(a);
if (b) buffer->append(b); if (b) buffer->append(b);

View file

@ -152,10 +152,13 @@ static const input_function_metadata_t input_function_metadata[] = {
{readline_cmd_t::func_or, L"or"}, {readline_cmd_t::func_or, L"or"},
{readline_cmd_t::expand_abbr, L"expand-abbr"}, {readline_cmd_t::expand_abbr, L"expand-abbr"},
{readline_cmd_t::delete_or_exit, L"delete-or-exit"}, {readline_cmd_t::delete_or_exit, L"delete-or-exit"},
{readline_cmd_t::exit, L"exit"},
{readline_cmd_t::cancel_commandline, L"cancel-commandline"}, {readline_cmd_t::cancel_commandline, L"cancel-commandline"},
{readline_cmd_t::cancel, L"cancel"}, {readline_cmd_t::cancel, L"cancel"},
{readline_cmd_t::undo, L"undo"}, {readline_cmd_t::undo, L"undo"},
{readline_cmd_t::redo, L"redo"}, {readline_cmd_t::redo, L"redo"},
{readline_cmd_t::begin_undo_group, L"begin-undo-group"},
{readline_cmd_t::end_undo_group, L"end-undo-group"},
}; };
static_assert(sizeof(input_function_metadata) / sizeof(input_function_metadata[0]) == static_assert(sizeof(input_function_metadata) / sizeof(input_function_metadata[0]) ==

View file

@ -75,10 +75,13 @@ enum class readline_cmd_t {
func_or, func_or,
expand_abbr, expand_abbr,
delete_or_exit, delete_or_exit,
exit,
cancel_commandline, cancel_commandline,
cancel, cancel,
undo, undo,
redo, redo,
begin_undo_group,
end_undo_group,
repeat_jump, repeat_jump,
// NOTE: This one has to be last. // NOTE: This one has to be last.
reverse_repeat_jump reverse_repeat_jump

View file

@ -94,7 +94,7 @@ void io_buffer_t::begin_filling(autoclose_fd_t fd) {
// We want to fill buffer_ by reading from fd. fd is the read end of a pipe; the write end is // We want to fill buffer_ by reading from fd. fd is the read end of a pipe; the write end is
// owned by another process, or something else writing in fish. // owned by another process, or something else writing in fish.
// Pass fd to an fd_monitor. It will add fd to its select() loop, and give us a callback when // Pass fd to an fd_monitor. It will add fd to its select() loop, and give us a callback when
// the fd is readable, or when our timeout is hit. The usual path is that we will get called // the fd is readable, or when our item is poked. The usual path is that we will get called
// back, read a bit from the fd, and append it to the buffer. Eventually the write end of the // back, read a bit from the fd, and append it to the buffer. Eventually the write end of the
// pipe will be closed - probably the other process exited - and fd will be widowed; read() will // pipe will be closed - probably the other process exited - and fd will be widowed; read() will
// then return 0 and we will stop reading. // then return 0 and we will stop reading.
@ -102,9 +102,10 @@ void io_buffer_t::begin_filling(autoclose_fd_t fd) {
// e.g.: // e.g.:
// cmd ( background & ; echo hi ) // cmd ( background & ; echo hi )
// Here the background process will inherit the write end of the pipe and hold onto it forever. // Here the background process will inherit the write end of the pipe and hold onto it forever.
// In this case, we will hit the timeout on waiting for more data and notice that the shutdown // In this case, when complete_background_fillthread() is called, the callback will be invoked
// flag is set (this indicates that the command substitution is done); in this case we will read // with item_wake_reason_t::poke, and we will notice that the shutdown flag is set (this
// until we get EAGAIN and then give up. // indicates that the command substitution is done); in this case we will read until we get
// EAGAIN and then give up.
// Construct a promise that can go into our background thread. // Construct a promise that can go into our background thread.
auto promise = std::make_shared<std::promise<void>>(); auto promise = std::make_shared<std::promise<void>>();
@ -113,25 +114,18 @@ void io_buffer_t::begin_filling(autoclose_fd_t fd) {
// Note this should only ever be called once. // Note this should only ever be called once.
fillthread_waiter_ = promise->get_future(); fillthread_waiter_ = promise->get_future();
// 100 msec poll rate. Note that in most cases, the write end of the pipe will be closed so
// select() will return; the polling is important only for weird cases like a background process
// launched in a command substitution.
constexpr uint64_t usec_per_msec = 1000;
uint64_t poll_usec = 100 * usec_per_msec;
// Run our function to read until the receiver is closed. // Run our function to read until the receiver is closed.
// It's OK to capture 'this' by value because 'this' waits for the promise in its dtor. // It's OK to capture 'this' by value because 'this' waits for the promise in its dtor.
fd_monitor_item_t item; fd_monitor_item_t item;
item.fd = std::move(fd); item.fd = std::move(fd);
item.timeout_usec = poll_usec; item.callback = [this, promise](autoclose_fd_t &fd, item_wake_reason_t reason) {
item.callback = [this, promise](autoclose_fd_t &fd, bool timed_out) {
ASSERT_IS_BACKGROUND_THREAD(); ASSERT_IS_BACKGROUND_THREAD();
// Only check the shutdown flag if we timed out. // Only check the shutdown flag if we timed out or were poked.
// It's important that if select() indicated we were readable, that we call select() again // It's important that if select() indicated we were readable, that we call select() again
// allowing it to time out. Note the typical case is that the fd will be closed, in which // allowing it to time out. Note the typical case is that the fd will be closed, in which
// case select will return immediately. // case select will return immediately.
bool done = false; bool done = false;
if (!timed_out) { if (reason == item_wake_reason_t::readable) {
// select() reported us as readable; read a bit. // select() reported us as readable; read a bit.
scoped_lock locker(append_lock_); scoped_lock locker(append_lock_);
ssize_t ret = read_once(fd.fd()); ssize_t ret = read_once(fd.fd());
@ -151,13 +145,16 @@ void io_buffer_t::begin_filling(autoclose_fd_t fd) {
promise->set_value(); promise->set_value();
} }
}; };
fd_monitor().add(std::move(item)); this->item_id_ = fd_monitor().add(std::move(item));
} }
void io_buffer_t::complete_background_fillthread() { void io_buffer_t::complete_background_fillthread() {
// Mark that our fillthread is done, then wake it up.
ASSERT_IS_MAIN_THREAD(); ASSERT_IS_MAIN_THREAD();
assert(fillthread_running() && "Should have a fillthread"); assert(fillthread_running() && "Should have a fillthread");
assert(this->item_id_ > 0 && "Should have a valid item ID");
shutdown_fillthread_ = true; shutdown_fillthread_ = true;
fd_monitor().poke_item(this->item_id_);
// Wait for the fillthread to fulfill its promise, and then clear the future so we know we no // Wait for the fillthread to fulfill its promise, and then clear the future so we know we no
// longer have one. // longer have one.

View file

@ -312,6 +312,9 @@ class io_buffer_t {
/// running. The fillthread fulfills the corresponding promise when it exits. /// running. The fillthread fulfills the corresponding promise when it exits.
std::future<void> fillthread_waiter_{}; std::future<void> fillthread_waiter_{};
/// The item id of our background fillthread fd monitor item.
uint64_t item_id_{0};
/// Lock for appending. /// Lock for appending.
std::mutex append_lock_{}; std::mutex append_lock_{};

View file

@ -131,7 +131,8 @@ struct thread_pool_t {
/// The thread pool for "iothreads" which are used to lift I/O off of the main thread. /// The thread pool for "iothreads" which are used to lift I/O off of the main thread.
/// These are used for completions, etc. /// These are used for completions, etc.
static thread_pool_t s_io_thread_pool(1, IO_MAX_THREADS); /// Leaked to avoid shutdown dtor registration (including tsan).
static thread_pool_t &s_io_thread_pool = *(new thread_pool_t(1, IO_MAX_THREADS));
static owning_lock<std::queue<void_function_t>> s_result_queue; static owning_lock<std::queue<void_function_t>> s_result_queue;

View file

@ -60,8 +60,9 @@ class operation_context_t {
size_t expansion_limit = kExpansionLimitDefault); size_t expansion_limit = kExpansionLimitDefault);
/// Construct from vars alone. /// Construct from vars alone.
explicit operation_context_t(const environment_t &vars) explicit operation_context_t(const environment_t &vars,
: operation_context_t(nullptr, vars, no_cancel) {} size_t expansion_limit = kExpansionLimitDefault)
: operation_context_t(nullptr, vars, no_cancel, expansion_limit) {}
~operation_context_t(); ~operation_context_t();
}; };

View file

@ -144,47 +144,24 @@ bool outputter_t::write_color(rgb_color_t color, bool is_fg) {
/// - Lastly we may need to write set_a_background or set_a_foreground to set the other half of the /// - Lastly we may need to write set_a_background or set_a_foreground to set the other half of the
/// color pair to what it should be. /// color pair to what it should be.
/// ///
/// \param c Foreground color. /// \param fg Foreground color.
/// \param c2 Background color. /// \param bg Background color.
void outputter_t::set_color(rgb_color_t c, rgb_color_t c2) { void outputter_t::set_color(rgb_color_t fg, rgb_color_t bg) {
if (!cur_term) return; // Test if we have at least basic support for setting fonts, colors and related bits - otherwise
// just give up...
if (!cur_term || !exit_attribute_mode) return;
const rgb_color_t normal = rgb_color_t::normal(); const rgb_color_t normal = rgb_color_t::normal();
bool bg_set = false, last_bg_set = false; bool bg_set = false, last_bg_set = false;
bool is_bold = false; bool is_bold = fg.is_bold() || bg.is_bold();
bool is_underline = false; bool is_underline = fg.is_underline() || bg.is_underline();
bool is_italics = false; bool is_italics = fg.is_italics() || bg.is_italics();
bool is_dim = false; bool is_dim = fg.is_dim() || bg.is_dim();
bool is_reverse = false; bool is_reverse = fg.is_reverse() || bg.is_reverse();
// Test if we have at least basic support for setting fonts, colors and related bits - otherwise if (fg.is_reset() || bg.is_reset()) {
// just give up... fg = bg = normal;
if (!exit_attribute_mode) { reset_modes();
return;
}
is_bold |= c.is_bold();
is_bold |= c2.is_bold();
is_underline |= c.is_underline();
is_underline |= c2.is_underline();
is_italics |= c.is_italics();
is_italics |= c2.is_italics();
is_dim |= c.is_dim();
is_dim |= c2.is_dim();
is_reverse |= c.is_reverse();
is_reverse |= c2.is_reverse();
if (c.is_reset() || c2.is_reset()) {
c = c2 = normal;
was_bold = false;
was_underline = false;
was_italics = false;
was_dim = false;
was_reverse = false;
// If we exit attibute mode, we must first set a color, or previously colored text might // If we exit attibute mode, we must first set a color, or previously colored text might
// lose it's color. Terminals are weird... // lose it's color. Terminals are weird...
write_foreground_color(*this, 0); write_foreground_color(*this, 0);
@ -192,40 +169,14 @@ void outputter_t::set_color(rgb_color_t c, rgb_color_t c2) {
return; return;
} }
if (was_bold && !is_bold) { if ((was_bold && !is_bold)
// Only way to exit bold mode is a reset of all attributes. || (was_dim && !is_dim)
|| (was_reverse && !is_reverse)) {
// Only way to exit bold/dim/reverse mode is a reset of all attributes.
writembs(*this, exit_attribute_mode); writembs(*this, exit_attribute_mode);
last_color = normal; last_color = normal;
last_color2 = normal; last_color2 = normal;
was_bold = false; reset_modes();
was_underline = false;
was_italics = false;
was_dim = false;
was_reverse = false;
}
if (was_dim && !is_dim) {
// Only way to exit dim mode is a reset of all attributes.
writembs(*this, exit_attribute_mode);
last_color = normal;
last_color2 = normal;
was_bold = false;
was_underline = false;
was_italics = false;
was_dim = false;
was_reverse = false;
}
if (was_reverse && !is_reverse) {
// Only way to exit reverse mode is a reset of all attributes.
writembs(*this, exit_attribute_mode);
last_color = normal;
last_color2 = normal;
was_bold = false;
was_underline = false;
was_italics = false;
was_dim = false;
was_reverse = false;
} }
if (!last_color2.is_normal() && !last_color2.is_reset()) { if (!last_color2.is_normal() && !last_color2.is_reset()) {
@ -233,10 +184,10 @@ void outputter_t::set_color(rgb_color_t c, rgb_color_t c2) {
last_bg_set = true; last_bg_set = true;
} }
if (!c2.is_normal()) { if (!bg.is_normal()) {
// Background is set. // Background is set.
bg_set = true; bg_set = true;
if (c == c2) c = (c2 == rgb_color_t::white()) ? rgb_color_t::black() : rgb_color_t::white(); if (fg == bg) fg = (bg == rgb_color_t::white()) ? rgb_color_t::black() : rgb_color_t::white();
} }
if (enter_bold_mode && enter_bold_mode[0] != '\0') { if (enter_bold_mode && enter_bold_mode[0] != '\0') {
@ -248,11 +199,7 @@ void outputter_t::set_color(rgb_color_t c, rgb_color_t c2) {
if (!bg_set && last_bg_set) { if (!bg_set && last_bg_set) {
// Background color changed and is no longer set, so we exit bold mode. // Background color changed and is no longer set, so we exit bold mode.
writembs(*this, exit_attribute_mode); writembs(*this, exit_attribute_mode);
was_bold = false; reset_modes();
was_underline = false;
was_italics = false;
was_dim = false;
was_reverse = false;
// We don't know if exit_attribute_mode resets colors, so we set it to something known. // We don't know if exit_attribute_mode resets colors, so we set it to something known.
if (write_foreground_color(*this, 0)) { if (write_foreground_color(*this, 0)) {
last_color = rgb_color_t::black(); last_color = rgb_color_t::black();
@ -260,26 +207,22 @@ void outputter_t::set_color(rgb_color_t c, rgb_color_t c2) {
} }
} }
if (last_color != c) { if (last_color != fg) {
if (c.is_normal()) { if (fg.is_normal()) {
write_foreground_color(*this, 0); write_foreground_color(*this, 0);
writembs(*this, exit_attribute_mode); writembs(*this, exit_attribute_mode);
last_color2 = rgb_color_t::normal(); last_color2 = rgb_color_t::normal();
was_bold = false; reset_modes();
was_underline = false; } else if (!fg.is_special()) {
was_italics = false; write_color(fg, true /* foreground */);
was_dim = false;
was_reverse = false;
} else if (!c.is_special()) {
write_color(c, true /* foreground */);
} }
} }
last_color = c; last_color = fg;
if (last_color2 != c2) { if (last_color2 != bg) {
if (c2.is_normal()) { if (bg.is_normal()) {
write_background_color(*this, 0); write_background_color(*this, 0);
writembs(*this, exit_attribute_mode); writembs(*this, exit_attribute_mode);
@ -287,15 +230,11 @@ void outputter_t::set_color(rgb_color_t c, rgb_color_t c2) {
write_color(last_color, true /* foreground */); write_color(last_color, true /* foreground */);
} }
was_bold = false; reset_modes();
was_underline = false; last_color2 = bg;
was_italics = false; } else if (!bg.is_special()) {
was_dim = false; write_color(bg, false /* not foreground */);
was_reverse = false; last_color2 = bg;
last_color2 = c2;
} else if (!c2.is_special()) {
write_color(c2, false /* not foreground */);
last_color2 = c2;
} }
} }

View file

@ -32,6 +32,14 @@ class outputter_t {
bool was_dim = false; bool was_dim = false;
bool was_reverse = false; bool was_reverse = false;
void reset_modes() {
was_bold = false;
was_underline = false;
was_italics = false;
was_dim = false;
was_reverse = false;
}
/// Construct an outputter which outputs to a given fd. /// Construct an outputter which outputs to a given fd.
explicit outputter_t(int fd) : fd_(fd) {} explicit outputter_t(int fd) : fd_(fd) {}

View file

@ -200,7 +200,7 @@ static bool want_to_coalesce_insertion_of(const editable_line_t &el, const wcstr
// Only consolidate single character inserts. // Only consolidate single character inserts.
if (str.size() != 1) return false; if (str.size() != 1) return false;
// Make an undo group after every space. // Make an undo group after every space.
if (str.at(0) == L' ') return false; if (str.at(0) == L' ' && !el.undo_history.try_coalesce) return false;
assert(!el.undo_history.edits.empty()); assert(!el.undo_history.edits.empty());
const edit_t &last_edit = el.undo_history.edits.back(); const edit_t &last_edit = el.undo_history.edits.back();
// Don't add to the last edit if it deleted something. // Don't add to the last edit if it deleted something.
@ -211,19 +211,35 @@ static bool want_to_coalesce_insertion_of(const editable_line_t &el, const wcstr
} }
bool editable_line_t::undo() { bool editable_line_t::undo() {
if (undo_history.edits_applied == 0) return false; // nothing to undo bool did_undo = false;
const edit_t &edit = undo_history.edits.at(undo_history.edits_applied - 1); maybe_t<int> last_group_id{-1};
undo_history.edits_applied--; while (undo_history.edits_applied != 0) {
edit_t inverse = edit_t(edit.offset, edit.replacement.size(), L""); const edit_t &edit = undo_history.edits.at(undo_history.edits_applied - 1);
inverse.replacement = edit.old; if (did_undo && (!edit.group_id.has_value() || edit.group_id != last_group_id)) {
size_t old_position = edit.cursor_position_before_edit; // We've restored all the edits in this logical undo group
apply_edit(&text_, inverse); break;
set_position(old_position); }
last_group_id = edit.group_id;
undo_history.edits_applied--;
edit_t inverse = edit_t(edit.offset, edit.replacement.size(), L"");
inverse.replacement = edit.old;
size_t old_position = edit.cursor_position_before_edit;
apply_edit(&text_, inverse);
set_position(old_position);
did_undo = true;
}
end_edit_group();
undo_history.may_coalesce = false; undo_history.may_coalesce = false;
return true; return did_undo;
} }
void editable_line_t::push_edit(edit_t &&edit) { void editable_line_t::push_edit(edit_t &&edit) {
// Assign a new group id or propagate the old one if we're in a logical grouping of edits
if (edit_group_level_ != -1) {
edit.group_id = edit_group_id_;
}
bool edit_does_nothing = edit.length == 0 && edit.replacement.empty(); bool edit_does_nothing = edit.length == 0 && edit.replacement.empty();
if (edit_does_nothing) return; if (edit_does_nothing) return;
if (undo_history.edits_applied != undo_history.edits.size()) { if (undo_history.edits_applied != undo_history.edits.size()) {
@ -251,13 +267,48 @@ void editable_line_t::insert_coalesce(const wcstring &str) {
} }
bool editable_line_t::redo() { bool editable_line_t::redo() {
if (undo_history.edits_applied >= undo_history.edits.size()) return false; // nothing to redo bool did_redo = false;
const edit_t &edit = undo_history.edits.at(undo_history.edits_applied);
undo_history.edits_applied++; maybe_t<int> last_group_id{-1};
apply_edit(&text_, edit); while (undo_history.edits_applied < undo_history.edits.size()) {
set_position(cursor_position_after_edit(edit)); const edit_t &edit = undo_history.edits.at(undo_history.edits_applied);
undo_history.may_coalesce = false; // Make a new undo group here. if (did_redo && (!edit.group_id.has_value() || edit.group_id != last_group_id)) {
return true; // We've restored all the edits in this logical undo group
break;
}
last_group_id = edit.group_id;
undo_history.edits_applied++;
apply_edit(&text_, edit);
set_position(cursor_position_after_edit(edit));
did_redo = true;
}
end_edit_group();
return did_redo;
}
void editable_line_t::begin_edit_group() {
if (++edit_group_level_ == 0) {
// Indicate that the next change must trigger the creation of a new history item
undo_history.may_coalesce = false;
// Indicate that future changes should be coalesced into the same edit if possible.
undo_history.try_coalesce = true;
// Assign a logical edit group id to future edits in this group
edit_group_id_ += 1;
}
}
void editable_line_t::end_edit_group() {
if (edit_group_level_ == -1) {
// Clamp the minimum value to -1 to prevent unbalanced end_edit_group() calls from breaking
// everything.
return;
}
if (--edit_group_level_ == -1) {
undo_history.try_coalesce = false;
undo_history.may_coalesce = false;
}
} }
namespace { namespace {
@ -395,7 +446,7 @@ class reader_history_search_t {
bool add_skip(const wcstring &str) { return skips_.insert(str).second; } bool add_skip(const wcstring &str) { return skips_.insert(str).second; }
/// Reset, beginning a new line or token mode search. /// Reset, beginning a new line or token mode search.
void reset_to_mode(const wcstring &text, history_t *hist, mode_t mode) { void reset_to_mode(const wcstring &text, const std::shared_ptr<history_t> &hist, mode_t mode) {
assert(mode != inactive && "mode cannot be inactive in this setter"); assert(mode != inactive && "mode cannot be inactive in this setter");
skips_ = {text}; skips_ = {text};
matches_ = {text}; matches_ = {text};
@ -407,7 +458,7 @@ class reader_history_search_t {
if (low == text) flags |= history_search_ignore_case; if (low == text) flags |= history_search_ignore_case;
// We can skip dedup in history_search_t because we do it ourselves in skips_. // We can skip dedup in history_search_t because we do it ourselves in skips_.
search_ = history_search_t( search_ = history_search_t(
*hist, text, hist, text,
by_prefix() ? history_search_type_t::prefix : history_search_type_t::contains, flags); by_prefix() ? history_search_type_t::prefix : history_search_type_t::contains, flags);
} }
@ -535,7 +586,7 @@ class reader_data_t : public std::enable_shared_from_this<reader_data_t> {
/// The source of input events. /// The source of input events.
inputter_t inputter; inputter_t inputter;
/// The history. /// The history.
history_t *history{nullptr}; std::shared_ptr<history_t> history{};
/// The history search. /// The history search.
reader_history_search_t history_search{}; reader_history_search_t history_search{};
@ -629,11 +680,12 @@ class reader_data_t : public std::enable_shared_from_this<reader_data_t> {
/// Access the parser. /// Access the parser.
parser_t &parser() { return *parser_ref; } parser_t &parser() { return *parser_ref; }
reader_data_t(std::shared_ptr<parser_t> parser, history_t *hist, reader_config_t &&conf) reader_data_t(std::shared_ptr<parser_t> parser, std::shared_ptr<history_t> hist,
reader_config_t &&conf)
: conf(std::move(conf)), : conf(std::move(conf)),
parser_ref(std::move(parser)), parser_ref(std::move(parser)),
inputter(*parser_ref, conf.in), inputter(*parser_ref, conf.in),
history(hist) {} history(std::move(hist)) {}
void update_buff_pos(editable_line_t *el, maybe_t<size_t> new_pos = none_t()); void update_buff_pos(editable_line_t *el, maybe_t<size_t> new_pos = none_t());
@ -643,8 +695,7 @@ class reader_data_t : public std::enable_shared_from_this<reader_data_t> {
/// Erase @length characters starting at @offset. /// Erase @length characters starting at @offset.
void erase_substring(editable_line_t *el, size_t offset, size_t length); void erase_substring(editable_line_t *el, size_t offset, size_t length);
/// Replace the text of length @length at @offset by @replacement. /// Replace the text of length @length at @offset by @replacement.
void replace_substring(editable_line_t *el, size_t offset, size_t length, void replace_substring(editable_line_t *el, size_t offset, size_t length, wcstring replacement);
const wcstring &replacement);
void push_edit(editable_line_t *el, edit_t &&edit); void push_edit(editable_line_t *el, edit_t &&edit);
/// Insert the character into the command line buffer and print it to the screen using syntax /// Insert the character into the command line buffer and print it to the screen using syntax
@ -1416,7 +1467,7 @@ void reader_data_t::insert_string(editable_line_t *el, const wcstring &str) {
assert(el->undo_history.may_coalesce); assert(el->undo_history.may_coalesce);
} else { } else {
el->push_edit(edit_t(el->position(), 0, str)); el->push_edit(edit_t(el->position(), 0, str));
el->undo_history.may_coalesce = (str.size() == 1); el->undo_history.may_coalesce = el->undo_history.try_coalesce || (str.size() == 1);
} }
if (el == &command_line) suppress_autosuggestion = false; if (el == &command_line) suppress_autosuggestion = false;
@ -1440,8 +1491,8 @@ void reader_data_t::erase_substring(editable_line_t *el, size_t offset, size_t l
} }
void reader_data_t::replace_substring(editable_line_t *el, size_t offset, size_t length, void reader_data_t::replace_substring(editable_line_t *el, size_t offset, size_t length,
const wcstring &replacement) { wcstring replacement) {
push_edit(el, edit_t(offset, length, replacement)); push_edit(el, edit_t(offset, length, std::move(replacement)));
} }
/// Insert the string in the given command line at the given cursor position. The function checks if /// Insert the string in the given command line at the given cursor position. The function checks if
@ -1577,14 +1628,11 @@ void reader_data_t::completion_insert(const wcstring &val, size_t token_end,
set_buffer_maintaining_pager(new_command_line, cursor); set_buffer_maintaining_pager(new_command_line, cursor);
} }
static bool may_add_to_history(const wcstring &commandline_prefix) {
return !commandline_prefix.empty() && commandline_prefix.at(0) != L' ';
}
// Returns a function that can be invoked (potentially // Returns a function that can be invoked (potentially
// on a background thread) to determine the autosuggestion // on a background thread) to determine the autosuggestion
static std::function<autosuggestion_t(void)> get_autosuggestion_performer( static std::function<autosuggestion_t(void)> get_autosuggestion_performer(
parser_t &parser, const wcstring &search_string, size_t cursor_pos, history_t *history) { parser_t &parser, const wcstring &search_string, size_t cursor_pos,
const std::shared_ptr<history_t> &history) {
const uint32_t generation_count = read_generation_count(); const uint32_t generation_count = read_generation_count();
auto vars = parser.vars().snapshot(); auto vars = parser.vars().snapshot();
const wcstring working_directory = vars->get_pwd_slash(); const wcstring working_directory = vars->get_pwd_slash();
@ -1604,21 +1652,20 @@ static std::function<autosuggestion_t(void)> get_autosuggestion_performer(
return nothing; return nothing;
} }
if (may_add_to_history(search_string)) { // Search history for a matching item.
history_search_t searcher(*history, search_string, history_search_type_t::prefix, history_search_t searcher(history.get(), search_string, history_search_type_t::prefix,
history_search_flags_t{}); history_search_flags_t{});
while (!ctx.check_cancel() && searcher.go_backwards()) { while (!ctx.check_cancel() && searcher.go_backwards()) {
const history_item_t &item = searcher.current_item(); const history_item_t &item = searcher.current_item();
// Skip items with newlines because they make terrible autosuggestions. // Skip items with newlines because they make terrible autosuggestions.
if (item.str().find(L'\n') != wcstring::npos) continue; if (item.str().find(L'\n') != wcstring::npos) continue;
if (autosuggest_validate_from_history(item, working_directory, ctx)) { if (autosuggest_validate_from_history(item, working_directory, ctx)) {
// The command autosuggestion was handled specially, so we're done. // The command autosuggestion was handled specially, so we're done.
// History items are case-sensitive, see #3978. // History items are case-sensitive, see #3978.
return autosuggestion_t{searcher.current_string(), search_string, return autosuggestion_t{searcher.current_string(), search_string,
false /* icase */}; false /* icase */};
}
} }
} }
@ -2467,7 +2514,7 @@ void reader_change_history(const wcstring &name) {
reader_data_t *data = current_data_or_null(); reader_data_t *data = current_data_or_null();
if (data && data->history) { if (data && data->history) {
data->history->save(); data->history->save();
data->history = &history_t::history_with_name(name); data->history = history_t::with_name(name);
} }
} }
@ -2476,7 +2523,7 @@ void reader_change_history(const wcstring &name) {
static std::shared_ptr<reader_data_t> reader_push_ret(parser_t &parser, static std::shared_ptr<reader_data_t> reader_push_ret(parser_t &parser,
const wcstring &history_name, const wcstring &history_name,
reader_config_t &&conf) { reader_config_t &&conf) {
history_t *hist = &history_t::history_with_name(history_name); std::shared_ptr<history_t> hist = history_t::with_name(history_name);
auto data = std::make_shared<reader_data_t>(parser.shared(), hist, std::move(conf)); auto data = std::make_shared<reader_data_t>(parser.shared(), hist, std::move(conf));
reader_data_stack.push_back(data); reader_data_stack.push_back(data);
data->command_line_changed(&data->command_line); data->command_line_changed(&data->command_line);
@ -3082,6 +3129,13 @@ void reader_data_t::handle_readline_command(readline_cmd_t c, readline_loop_stat
delete_char(); delete_char();
break; break;
} }
case rl::exit: {
// This is by definition a successful exit, override the status
parser().set_last_statuses(statuses_t::just(STATUS_CMD_OK));
exit_loop_requested = true;
check_exit_loop_maybe_warning(this);
break;
}
case rl::delete_or_exit: case rl::delete_or_exit:
case rl::delete_char: { case rl::delete_char: {
// Remove the current character in the character buffer and on the screen using // Remove the current character in the character buffer and on the screen using
@ -3159,12 +3213,40 @@ void reader_data_t::handle_readline_command(readline_cmd_t c, readline_loop_stat
} }
if (command_test_result == 0) { if (command_test_result == 0) {
// Finished command, execute it. Don't add items that start with a leading // Finished command, execute it. Don't add items in silent mode (#7230).
// space, or if in silent mode (#7230). wcstring text = command_line.text();
const editable_line_t *el = &command_line; if (text.empty()) {
if (history != nullptr && !conf.in_silent_mode && may_add_to_history(el->text())) { // Here the user just hit return. Make a new prompt, don't remove ephemeral
history->add_pending_with_file_detection(el->text(), vars.get_pwd_slash()); // items.
rls.finished = true;
break;
} }
// Historical behavior is to trim trailing spaces.
while (!text.empty() && text.back() == L' ') {
text.pop_back();
}
if (history && !conf.in_silent_mode) {
// Remove ephemeral items.
// Note we fall into this case if the user just types a space and hits return.
history->remove_ephemeral_items();
// Mark this item as ephemeral if there is a leading space (#615).
history_persistence_mode_t mode;
if (text.front() == L' ') {
// Leading spaces are ephemeral (#615).
mode = history_persistence_mode_t::ephemeral;
} else if (in_private_mode(vars)) {
// Private mode means in-memory only.
mode = history_persistence_mode_t::memory;
} else {
mode = history_persistence_mode_t::disk;
}
history_t::add_pending_with_file_detection(history, text, vars.snapshot(),
mode);
}
rls.finished = true; rls.finished = true;
update_buff_pos(&command_line, command_line.size()); update_buff_pos(&command_line, command_line.size());
} else if (command_test_result == PARSER_TEST_INCOMPLETE) { } else if (command_test_result == PARSER_TEST_INCOMPLETE) {
@ -3682,7 +3764,17 @@ void reader_data_t::handle_readline_command(readline_cmd_t c, readline_loop_stat
} }
break; break;
} }
// Some commands should have been handled internally by inputter_t::readch(). case rl::begin_undo_group: {
editable_line_t *el = active_edit_line();
el->begin_edit_group();
break;
}
case rl::end_undo_group: {
editable_line_t *el = active_edit_line();
el->end_edit_group();
break;
}
// Some commands should have been handled internally by inputter_t::readch().
case rl::self_insert: case rl::self_insert:
case rl::self_insert_notfirst: case rl::self_insert_notfirst:
case rl::func_or: case rl::func_or:
@ -3953,6 +4045,13 @@ void reader_schedule_prompt_repaint() {
} }
} }
void reader_handle_command(readline_cmd_t cmd) {
if (reader_data_t *data = current_data_or_null()) {
readline_loop_state_t rls{};
data->handle_readline_command(cmd, rls);
}
}
void reader_queue_ch(const char_event_t &ch) { void reader_queue_ch(const char_event_t &ch) {
if (reader_data_t *data = current_data_or_null()) { if (reader_data_t *data = current_data_or_null()) {
data->inputter.queue_ch(ch); data->inputter.queue_ch(ch);
@ -3965,7 +4064,7 @@ const wchar_t *reader_get_buffer() {
return data ? data->command_line.text().c_str() : nullptr; return data ? data->command_line.text().c_str() : nullptr;
} }
history_t *reader_get_history() { std::shared_ptr<history_t> reader_get_history() {
ASSERT_IS_MAIN_THREAD(); ASSERT_IS_MAIN_THREAD();
reader_data_t *data = current_data_or_null(); reader_data_t *data = current_data_or_null();
return data ? data->history : nullptr; return data ? data->history : nullptr;

View file

@ -31,6 +31,10 @@ struct edit_t {
/// The strings that are removed and added by this edit, respectively. /// The strings that are removed and added by this edit, respectively.
wcstring old, replacement; wcstring old, replacement;
/// edit_t is only for contiguous changes, so to restore a group of arbitrary changes to the
/// command line we need to have a group id as forcibly coalescing changes is not enough.
maybe_t<int> group_id;
explicit edit_t(size_t offset, size_t length, wcstring replacement) explicit edit_t(size_t offset, size_t length, wcstring replacement)
: offset(offset), length(length), replacement(std::move(replacement)) {} : offset(offset), length(length), replacement(std::move(replacement)) {}
@ -61,6 +65,11 @@ struct undo_history_t {
/// last one. /// last one.
bool may_coalesce = false; bool may_coalesce = false;
/// Whether to be more aggressive in coalescing edits. Ideally, it would be "force coalesce"
/// with guaranteed atomicity but as `edit_t` is strictly for contiguous changes, that guarantee
/// can't be made at this time.
bool try_coalesce = false;
/// Empty the history. /// Empty the history.
void clear(); void clear();
}; };
@ -72,6 +81,12 @@ class editable_line_t {
/// The current position of the cursor in the command line. /// The current position of the cursor in the command line.
size_t position_ = 0; size_t position_ = 0;
/// The nesting level for atomic edits, so that recursive invocations of start_edit_group()
/// are not ended by one end_edit_group() call.
int32_t edit_group_level_ = -1;
/// Monotonically increasing edit group, ignored when edit_group_level_ is -1. Allowed to wrap.
uint32_t edit_group_id_ = -1;
public: public:
undo_history_t undo_history; undo_history_t undo_history;
@ -110,6 +125,11 @@ class editable_line_t {
/// Redo the most recent undo. Returns true on success. /// Redo the most recent undo. Returns true on success.
bool redo(); bool redo();
/// Start a logical grouping of command line edits that should be undone/redone together.
void begin_edit_group();
/// End a logical grouping of command line edits that should be undone/redone together.
void end_edit_group();
}; };
/// Read commands from \c fd until encountering EOF. /// Read commands from \c fd until encountering EOF.
@ -149,7 +169,7 @@ void reader_queue_ch(const char_event_t &ch);
const wchar_t *reader_get_buffer(); const wchar_t *reader_get_buffer();
/// Returns the current reader's history. /// Returns the current reader's history.
history_t *reader_get_history(); std::shared_ptr<history_t> reader_get_history();
/// Set the string of characters in the command buffer, as well as the cursor position. /// Set the string of characters in the command buffer, as well as the cursor position.
/// ///

View file

@ -867,20 +867,14 @@ void wildcard_expander_t::expand_last_segment(const wcstring &base_dir, DIR *bas
/// wrappers around this one. /// wrappers around this one.
/// ///
/// This function traverses the relevant directory tree looking for matches, and recurses when /// This function traverses the relevant directory tree looking for matches, and recurses when
/// needed to handle wildcrards spanning multiple components and recursive wildcards. /// needed to handle wildcards spanning multiple components and recursive wildcards.
///
/// Because this function calls itself recursively with substrings, it's important that the
/// parameters be raw pointers instead of wcstring, which would be too expensive to construct for
/// all substrings.
/// ///
/// Args: /// Args:
/// base_dir: the "working directory" against which the wildcard is to be resolved /// base_dir: the "working directory" against which the wildcard is to be resolved
/// wc: the wildcard string itself, e.g. foo*bar/baz (where * is actually ANY_CHAR) /// wc: the wildcard string itself, e.g. foo*bar/baz (where * is actually ANY_CHAR)
/// prefix: the string that should be prepended for completions that replace their token. /// effective_prefix: the string that should be prepended for completions that replace their token.
// This is usually the same thing as the original wildcard, but for fuzzy matching, we /// This is usually the same thing as the original wildcard, but for fuzzy matching, we
// expand intermediate segments. effective_prefix is always either empty, or ends with a slash /// expand intermediate segments. effective_prefix is always either empty, or ends with a slash
// Note: this is only used when doing completions (for_completions is true), not
// expansions
void wildcard_expander_t::expand(const wcstring &base_dir, const wchar_t *wc, void wildcard_expander_t::expand(const wcstring &base_dir, const wchar_t *wc,
const wcstring &effective_prefix) { const wcstring &effective_prefix) {
assert(wc != nullptr); assert(wc != nullptr);
@ -890,10 +884,9 @@ void wildcard_expander_t::expand(const wcstring &base_dir, const wchar_t *wc,
} }
// Get the current segment and compute interesting properties about it. // Get the current segment and compute interesting properties about it.
const size_t wc_len = std::wcslen(wc);
const wchar_t *const next_slash = std::wcschr(wc, L'/'); const wchar_t *const next_slash = std::wcschr(wc, L'/');
const bool is_last_segment = (next_slash == nullptr); const bool is_last_segment = (next_slash == nullptr);
const size_t wc_segment_len = next_slash ? next_slash - wc : wc_len; const size_t wc_segment_len = next_slash ? next_slash - wc : std::wcslen(wc);
const wcstring wc_segment = wcstring(wc, wc_segment_len); const wcstring wc_segment = wcstring(wc, wc_segment_len);
const bool segment_has_wildcards = const bool segment_has_wildcards =
wildcard_has(wc_segment, true /* internal, i.e. look for ANY_CHAR instead of ? */); wildcard_has(wc_segment, true /* internal, i.e. look for ANY_CHAR instead of ? */);
@ -937,6 +930,20 @@ void wildcard_expander_t::expand(const wcstring &base_dir, const wchar_t *wc,
} }
} else { } else {
assert(!wc_segment.empty() && (segment_has_wildcards || is_last_segment)); assert(!wc_segment.empty() && (segment_has_wildcards || is_last_segment));
if (!is_last_segment && wc_segment == wcstring{ANY_STRING_RECURSIVE}) {
// Hack for #7222. This is an intermediate wc segment that is exactly **. The
// tail matches in subdirectories as normal, but also the current directory.
// That is, '**/bar' may match 'bar' and 'foo/bar'.
// Implement this by matching the wildcard tail only, in this directory.
// Note if the segment is not exactly ANY_STRING_RECURSIVE then the segment may only
// match subdirectories.
this->expand(base_dir, wc_remainder, effective_prefix);
if (interrupted_or_overflowed()) {
return;
}
}
DIR *dir = open_dir(base_dir); DIR *dir = open_dir(base_dir);
if (dir) { if (dir) {
if (is_last_segment) { if (is_last_segment) {
@ -949,9 +956,9 @@ void wildcard_expander_t::expand(const wcstring &base_dir, const wchar_t *wc,
effective_prefix + wc_segment + L'/'); effective_prefix + wc_segment + L'/');
} }
// Recursive wildcards require special handling.
size_t asr_idx = wc_segment.find(ANY_STRING_RECURSIVE); size_t asr_idx = wc_segment.find(ANY_STRING_RECURSIVE);
if (asr_idx != wcstring::npos) { if (asr_idx != wcstring::npos) {
// Apply the recursive **.
// Construct a "head + any" wildcard for matching stuff in this directory, and an // Construct a "head + any" wildcard for matching stuff in this directory, and an
// "any + tail" wildcard for matching stuff in subdirectories. Note that the // "any + tail" wildcard for matching stuff in subdirectories. Note that the
// ANY_STRING_RECURSIVE character is present in both the head and the tail. // ANY_STRING_RECURSIVE character is present in both the head and the tail.

View file

@ -24,11 +24,9 @@ enum {
/// Expand the wildcard by matching against the filesystem. /// Expand the wildcard by matching against the filesystem.
/// ///
/// New strings are allocated using malloc and should be freed by the caller.
///
/// wildcard_expand works by dividing the wildcard into segments at each directory boundary. Each /// wildcard_expand works by dividing the wildcard into segments at each directory boundary. Each
/// segment is processed separatly. All except the last segment are handled by matching the wildcard /// segment is processed separately. All except the last segment are handled by matching the
/// segment against all subdirectories of matching directories, and recursively calling /// wildcard segment against all subdirectories of matching directories, and recursively calling
/// wildcard_expand for matches. On the last segment, matching is made to any file, and all matches /// wildcard_expand for matches. On the last segment, matching is made to any file, and all matches
/// are inserted to the list. /// are inserted to the list.
/// ///

View file

@ -350,6 +350,54 @@ begin
# CHECK: saved_status 57 # CHECK: saved_status 57
end end
# long-only flags
begin
argparse installed= foo -- --installed=no --foo
set -l
# CHECK: _flag_a 'alpha' 'aaaa'
# CHECK: _flag_b -b
# CHECK: _flag_break -b
# CHECK: _flag_foo --foo
# CHECK: _flag_installed no
# CHECK: _flag_m 1
# CHECK: _flag_max 1
# CHECK: argv
# CHECK: saved_status 57
end
begin
argparse installed='!_validate_int --max 12' foo -- --installed=5 --foo
set -l
# CHECK: _flag_a 'alpha' 'aaaa'
# CHECK: _flag_b -b
# CHECK: _flag_break -b
# CHECK: _flag_foo --foo
# CHECK: _flag_installed 5
# CHECK: _flag_m 1
# CHECK: _flag_max 1
# CHECK: argv
# CHECK: saved_status 57
end
begin
argparse '#num' installed= -- --installed=5 -5
set -l
# CHECK: _flag_a 'alpha' 'aaaa'
# CHECK: _flag_b -b
# CHECK: _flag_break -b
# CHECK: _flag_installed 5
# CHECK: _flag_m 1
# CHECK: _flag_max 1
# CHECK: _flag_num 5
# CHECK: argv
# CHECK: saved_status 57
end
begin
argparse installed='!_validate_int --max 12' foo -- --foo --installed=error --foo
# CHECKERR: argparse: Value 'error' for flag 'installed' is not an integer
end
# #6483 - error messages for missing arguments # #6483 - error messages for missing arguments
argparse -n foo q r/required= -- foo -qr argparse -n foo q r/required= -- foo -qr
# CHECKERR: foo: Expected argument for option r # CHECKERR: foo: Expected argument for option r
@ -440,3 +488,12 @@ function wrongargparse
argparse a-b argparse a-b
argparse argparse
end end
begin
argparse ''
#CHECKERR: argparse: An option spec must have at least a short or a long flag
#CHECKERR: checks/argparse.fish (line {{\d+}}):
#CHECKERR: argparse ''
#CHECKERR: ^
#CHECKERR: (Type 'help argparse' for related documentation)
end

View file

@ -475,3 +475,12 @@ echo $status
builtin --query echo builtin --query echo
echo $status echo $status
#CHECK: 0 #CHECK: 0
# Check that echo doesn't interpret options *and print them*
# at the start of quoted args:
echo '-ne \tart'
# CHECK: -ne \tart
echo '-n art'
echo banana
# CHECK: -n art
# CHECK: banana

View file

@ -126,10 +126,73 @@ test $PWD = $base/-testdir
echo $status echo $status
#CHECK: 0 #CHECK: 0
# test a few error cases - nonexistent directory
set -l old_cdpath $CDPATH
set -l old_path $PWD
cd nonexistent
#CHECKERR: cd: The directory 'nonexistent' does not exist
#CHECKERR: {{.*}}/cd.fish (line {{\d+}}):
#CHECKERR: builtin cd $argv
#CHECKERR: ^
#CHECKERR: in function 'cd' with arguments 'nonexistent'
#CHECKERR: called on line {{\d+}} of file {{.*}}/cd.fish
touch file
cd file
#CHECKERR: cd: 'file' is not a directory
#CHECKERR: {{.*}}/cd.fish (line {{\d+}}):
#CHECKERR: builtin cd $argv
#CHECKERR: ^
#CHECKERR: in function 'cd' with arguments 'file'
#CHECKERR: called on line {{\d+}} of file {{.*}}/cd.fish
# a directory that isn't executable
mkdir bad-perms
chmod -x bad-perms
cd bad-perms
#CHECKERR: cd: Permission denied: 'bad-perms'
#CHECKERR: {{.*}}/cd.fish (line {{\d+}}):
#CHECKERR: builtin cd $argv
#CHECKERR: ^
#CHECKERR: in function 'cd' with arguments 'bad-perms'
#CHECKERR: called on line {{\d+}} of file {{.*}}/cd.fish
cd $old_path
mkdir -p cdpath-dir/bad-perms
mkdir -p cdpath-dir/nonexistent
mkdir -p cdpath-dir/file
set CDPATH $PWD/cdpath-dir $old_cdpath
# A different directory with the same name that is first in $CDPATH works.
cd bad-perms
cd $old_path
cd nonexistent
cd $old_path
cd file
cd $old_path
# Even if the good dirs are later in $CDPATH most errors still aren't a problem
# - they just cause us to keep looking.
cd $old_path
set CDPATH $old_cdpath $PWD/cdpath-dir
cd nonexistent
cd $old_path
cd bad-perms
# Permission errors are still a problem!
#CHECKERR: cd: Permission denied: 'bad-perms'
#CHECKERR: {{.*}}/cd.fish (line {{\d+}}):
#CHECKERR: builtin cd $argv
#CHECKERR: ^
#CHECKERR: in function 'cd' with arguments 'bad-perms'
#CHECKERR: called on line {{\d+}} of file {{.*}}/cd.fish
cd $old_path
cd file
cd $old_path
# cd back before removing the test directory again. # cd back before removing the test directory again.
cd $oldpwd cd $oldpwd
rm -Rf $base rm -Rf $base
set -g CDPATH ./
# Verify that PWD on-variable events are sent # Verify that PWD on-variable events are sent
function __fish_test_changed_pwd --on-variable PWD function __fish_test_changed_pwd --on-variable PWD

View file

@ -1,8 +1,25 @@
# RUN: %fish %s # RUN: %fish %s
# This tests various corner cases involving command substitution. Most # This tests various corner cases involving command substitution.
# importantly the limits on the amount of data we'll substitute.
# Test cmdsubs which spawn background processes - see #7559.
# If this hangs, it means that fish keeps trying to read from the write
# end of the cmdsub pipe (which has escaped).
# FIXME: we need to mark full job control for sleep to get its own pgid;
# otherwise $last_pid will return fish's pgid! It's always been so!
status job-control full
echo (command sleep 1000000 & ; set -g sleep_pid $last_pid ; echo local)
# CHECK: local
echo $sleep_pid
# CHECK: {{[1-9]\d*}}
kill $sleep_pid ; echo $status
# CHECK: 0
status job-control interactive
# Test limiting the amount of data we'll substitute.
set fish_read_limit 512 set fish_read_limit 512
function subme function subme
@ -10,7 +27,7 @@ function subme
echo $x echo $x
end end
# Command sub just under the limit should succeed # Command sub just under the limit should succeed.
set a (subme 511) set a (subme 511)
set --show a set --show a
#CHECK: $a: set in global scope, unexported, with 1 elements #CHECK: $a: set in global scope, unexported, with 1 elements

90
tests/checks/glob.fish Normal file
View file

@ -0,0 +1,90 @@
# RUN: %fish %s
set -l oldpwd $PWD
cd (mktemp -d)
set tmpdir (pwd -P)
# Hidden files are only matched with explicit dot.
touch .hidden visible
string join \n * | sort
# CHECK: visible
string join \n .* | sort
# CHECK: .hidden
rm -Rf .hidden visible
# Trailing slash matches only directories.
touch abc1
mkdir abc2
string join \n * | sort
# CHECK: abc1
# CHECK: abc2
string join \n */ | sort
# CHECK: abc2/
rm -Rf *
# Symlinks are descended into independently.
# Here dir2/link2 is symlinked to dir1/child1.
# The contents of dir2 will be explored twice.
mkdir -p dir1/child1
touch dir1/child1/anyfile
mkdir dir2
ln -s ../dir1/child1 dir2/link2
string join \n **/anyfile | sort
# CHECK: dir1/child1/anyfile
# CHECK: dir2/link2/anyfile
# But symlink loops only get explored once.
mkdir -p dir1/child2/grandchild1
touch dir1/child2/grandchild1/differentfile
ln -s ../../child2/grandchild1 dir1/child2/grandchild1/link2
echo **/differentfile
# CHECK: dir1/child2/grandchild1/differentfile
rm -Rf *
# Recursive globs handling.
mkdir -p dir_a1/dir_a2/dir_a3
touch dir_a1/dir_a2/dir_a3/file_a
mkdir -p dir_b1/dir_b2/dir_b3
touch dir_b1/dir_b2/dir_b3/file_b
string join \n **/file_* | sort
# CHECK: dir_a1/dir_a2/dir_a3/file_a
# CHECK: dir_b1/dir_b2/dir_b3/file_b
string join \n **a3/file_* | sort
# CHECK: dir_a1/dir_a2/dir_a3/file_a
string join \n ** | sort
# CHECK: dir_a1
# CHECK: dir_a1/dir_a2
# CHECK: dir_a1/dir_a2/dir_a3
# CHECK: dir_a1/dir_a2/dir_a3/file_a
# CHECK: dir_b1
# CHECK: dir_b1/dir_b2
# CHECK: dir_b1/dir_b2/dir_b3
# CHECK: dir_b1/dir_b2/dir_b3/file_b
string join \n **/ | sort
# CHECK: dir_a1/
# CHECK: dir_a1/dir_a2/
# CHECK: dir_a1/dir_a2/dir_a3/
# CHECK: dir_b1/
# CHECK: dir_b1/dir_b2/
# CHECK: dir_b1/dir_b2/dir_b3/
string join \n **a2/** | sort
# CHECK: dir_a1/dir_a2/dir_a3
# CHECK: dir_a1/dir_a2/dir_a3/file_a
rm -Rf *
# Special behavior for #7222.
# The literal segment ** matches in the same directory.
mkdir foo
touch bar foo/bar
string join \n **/bar | sort
# CHECK: bar
# CHECK: foo/bar
# Clean up.
cd $oldpwd
rm -Rf $tmpdir

View file

@ -53,3 +53,6 @@ $fish -c 'string escape y$argv' -c 'string escape x$argv' 1 2 3
# CHECK: x1 # CHECK: x1
# CHECK: x2 # CHECK: x2
# CHECK: x3 # CHECK: x3
# Should just do nothing.
$fish --no-execute

View file

@ -411,6 +411,27 @@ string repeat -n3 -m20 foo
string repeat -m4 foo string repeat -m4 foo
# CHECK: foof # CHECK: foof
string repeat -n 5 a b c
# CHECK: aaaaa
# CHECK: bbbbb
# CHECK: ccccc
string repeat -n 5 --max 4 123 456 789
# CHECK: 1231
# CHECK: 4564
# CHECK: 7897
string repeat -n 5 --max 4 123 '' 789
# CHECK: 1231
# CHECK:
# CHECK: 7897
# Historical string repeat behavior is no newline if no output.
echo -n before
string repeat -n 5 ''
echo after
# CHECK: beforeafter
string repeat -n-1 foo; and echo "exit 0" string repeat -n-1 foo; and echo "exit 0"
# CHECKERR: string repeat: Invalid count value '-1' # CHECKERR: string repeat: Invalid count value '-1'

View file

@ -20,6 +20,8 @@ set -e ITERM_PROFILE
# Test files specified on commandline, or all pexpect files. # Test files specified on commandline, or all pexpect files.
if set -q argv[1] if set -q argv[1]
set pexpect_files_to_test pexpects/$argv.py set pexpect_files_to_test pexpects/$argv.py
else if set -q FISH_PEXPECT_FILES
set pexpect_files_to_test (string replace -r '^.*/(?=pexpects/)' '' -- $FISH_PEXPECT_FILES)
else else
set pexpect_files_to_test pexpects/*.py set pexpect_files_to_test pexpects/*.py
end end

View file

@ -51,5 +51,16 @@ expect_prompt("hoge")
sendline("echo hoge >| \n cat") sendline("echo hoge >| \n cat")
expect_prompt("hoge") expect_prompt("hoge")
sendline("$fish --no-execute 2>&1")
expect_prompt("error: no-execute mode enabled and no script given. Exiting")
sendline("source; or echo failed") sendline("source; or echo failed")
expect_prompt("failed") expect_prompt("failed")
# See that `type` tells us the function was defined interactively.
sendline("function foo; end; type foo")
expect_str("foo is a function with definition\r\n")
expect_str("# Defined interactively\r\n")
expect_str("function foo")
expect_str("end")
expect_prompt()

View file

@ -0,0 +1,70 @@
#!/usr/bin/env python3
import os
import time
from pexpect_helper import SpawnedProc
sp = SpawnedProc()
sendline, sleep, expect_prompt, expect_str = (
sp.sendline,
sp.sleep,
sp.expect_prompt,
sp.expect_str,
)
# Helper to sendline and add to our view of history.
recorded_history = []
private_mode_active = False
fish_path = os.environ.get("fish")
# Send a line and record it in our history array if private mode is not active.
def sendline_record(s):
sendline(s)
if not private_mode_active:
recorded_history.append(s)
expect_prompt()
# Start off with no history.
sendline(r" builtin history clear; builtin history save")
expect_prompt()
# Ensure that fish_private_mode can be changed - see #7589.
sendline_record(r"echo before_private_mode")
expect_prompt("before_private_mode")
sendline(r" builtin history save")
expect_prompt()
# Enter private mode.
sendline_record(r"set -g fish_private_mode 1")
expect_prompt()
private_mode_active = True
sendline_record(r"echo check2 $fish_private_mode")
expect_prompt("check2 1")
# Nothing else gets added.
sendline_record(r"true")
expect_prompt()
sendline_record(r"false")
expect_prompt()
# Leave private mode. The command to leave it is still private.
sendline_record(r"set -ge fish_private_mode")
expect_prompt()
private_mode_active = False
# New commands get added.
sendline_record(r"set alpha beta")
expect_prompt()
# Check our history is what we expect.
# We have to wait for the time to tick over, else our item risks being discarded.
now = time.time()
start = int(now)
while now - start < 1:
sleep(now - start)
now = time.time()
sendline(r" builtin history save ; %s -c 'string join \n -- $history'" % fish_path)
expect_prompt("\r\n".join(reversed(recorded_history)))

34
tests/pexpects/undo.py Normal file
View file

@ -0,0 +1,34 @@
#!/usr/bin/env python3
from pexpect_helper import SpawnedProc
sp = SpawnedProc()
send, sendline, sleep, expect_prompt, expect_re, expect_str = (
sp.send,
sp.sendline,
sp.sleep,
sp.expect_prompt,
sp.expect_re,
sp.expect_str,
)
expect_prompt()
sendline("bind Undo undo; bind Redo redo")
expect_prompt()
send("echo word")
expect_str("echo word")
expect_str("echo word") # Not sure why we get this twice.
# FIXME why does this only undo one character? It undoes the entire word when run interactively.
send("Undo")
expect_str("echo wor")
send("Undo")
expect_str("echo ")
send("Redo")
expect_str("echo wor")
# FIXME see above.
send("Redo")
expect_str("echo word")

View file

@ -125,21 +125,6 @@ function say -V suppress_color
end end
end end
function colordiff -d 'Colored diff output for unified diffs'
diff $argv | while read -l line
switch $line
case '+*'
say green $line
case '-*'
say red $line
case '@*'
say cyan $line
case '*'
echo $line
end
end
end
# lame timer # lame timer
for program in {g,}date for program in {g,}date
if command -q $program && $program --version 1>/dev/null 2>/dev/null if command -q $program && $program --version 1>/dev/null 2>/dev/null