|
NAMEfish-doc - fish-shell DocumentationThis is the documentation for fish, the friendly interactive shell. A shell is a program that helps you operate your computer by starting other programs. fish offers a command-line interface focused on usability and interactive use. Unlike other shells, fish does not follow the POSIX standard, but still uses roughly the same model. Some of the special features of fish are:
This page explains how to install and set up fish and where to get more information. FURTHER READINGIf this is your first time using fish, see the tutorial.If you are already familiar with other shells like bash and want to see the scripting differences, see Fish For Bash Users. For a comprehensive overview of fish's scripting language, see The Fish Language. For information on using fish interactively, see Interactive use. INSTALLATION AND STARTThis section describes how to install, uninstall, start, and exit the fish shell. It also explains how to make fish the default shell.InstallationUp-to-date instructions for installing the latest version of fish are on the fish homepage.To install the development version of fish, see the instructions on the project's GitHub page. Starting and ExitingOnce fish has been installed, open a terminal. If fish is not the default shell:
> fish
> exit Executing BashIf fish is your default shell and you want to copy commands from the internet that are written in bash (the default shell on most systems), you can proceed in one of the following two ways:
> bash -c 'some bash command'
> bash $ some bash command $ exit > _ Default ShellTo make fish your default shell:
For detailed instructions see Switching to fish. UninstallingFor uninstalling fish: see FAQ: Uninstalling fish.Shebang LineBecause shell scripts are written in many different languages, they need to carry information about which interpreter should be used to execute them. For this, they are expected to have a first line, the shebang line, which names the interpreter executable.A script written in bash would need a first line like this: #!/bin/bash When the shell tells the kernel to execute the file, it will use the interpreter /bin/bash. For a script written in another language, just replace /bin/bash with the interpreter for that language (for example: /usr/bin/python for a python script, or /usr/local/bin/fish for a fish script). This line is only needed when scripts are executed without specifying the interpreter. For functions inside fish or when executing a script with fish /path/to/script, a shebang is not required (but it doesn't hurt!). CONFIGURATION FILESWhen fish is started, it reads and runs its configuration files. Where these are depends on build configuration and environment variables.The main file is ~/.config/fish/config.fish (or more precisely $XDG_CONFIG_HOME/fish/config.fish). Configuration files are evaluated in the following order:
If there are multiple files with the same name in these directories, only the first will be executed. They are executed in order of their filename, sorted (like globs) in a natural order (i.e. "01" sorts before "2").
~/.config/fish/config.fish is sourced after the snippets. This is so users can copy snippets and override some of their behavior. These files are all executed on the startup of every shell. If you want to run a command only on starting an interactive shell, use the exit status of the command status --is-interactive to determine if the shell is interactive. If you want to run a command only when using a login shell, use status --is-login instead. This will speed up the starting of non-interactive or non-login shells. If you are developing another program, you may wish to install configuration which is run for all users of the fish shell on a system. This is discouraged; if not carefully written, they may have side-effects or slow the startup of the shell. Additionally, users of other shells will not benefit from the Fish-specific configuration. However, if they are absolutely required, you may install them to the "vendor" configuration directory. As this path may vary from system to system, the pkgconfig framework should be used to discover this path with the output of pkg-config --variable confdir fish. Examples: If you want to add the directory ~/linux/bin to your PATH variable when using a login shell, add this to your ~/.config/fish/config.fish file: if status --is-login set -gx PATH $PATH ~/linux/bin end (alternatively use fish_add_path like fish_add_path ~/linux/bin, which only adds the path if it isn't included yet) If you want to run a set of commands when fish exits, use an event handler that is triggered by the exit of the shell: function on_exit --on-event fish_exit echo fish is now exiting end FURTHER HELP AND DEVELOPMENTIf you have a question not answered by this documentation, there are several avenues for help:
If you have an improvement for fish, you can submit it via the GitHub page. OTHER HELP PAGESFrequently asked questionsWhat is the equivalent to this thing from bash (or other shells)?See Fish for bash usersHow do I set or clear an environment variable?Use the set command:set -x key value # typically set -gx key value set -e key Since fish 3.1 you can set an environment variable for just one command using the key=value some command syntax, like in other shells. The two lines below behave identically - unlike other shells, fish will output value both times: key=value echo $key begin; set -lx key value; echo $key; end Note that "exported" is not a scope, but an additional bit of state. A variable can be global and exported or local and exported or even universal and exported. Typically it makes sense to make an exported variable global. How do I check whether a variable is defined?Use set -q var. For example, if set -q var; echo variable defined; end. To check multiple variables you can combine with and and or like so:if set -q var1; or set -q var2 echo either variable defined end Keep in mind that a defined variabled could also be empty, either by having no elements (if set like set var) or only empty elements (if set like set var ""). Read on for how to deal with those. How do I check whether a variable is not empty?Use string length -q -- $var. For example, if string length -q -- $var; echo not empty; end. Note that string length will interpret a list of multiple variables as a disjunction (meaning any/or):if string length -q -- $var1 $var2 $var3 echo at least one of these variables is not empty end Alternatively, use test -n "$var", but remember that the variable must be double-quoted. For example, if test -n "$var"; echo not empty; end. The test command provides its own and (-a) and or (-o): if test -n "$var1" -o -n "$var2" -o -n "$var3" echo at least one of these variables is not empty end If you want to know if a variable has no elements, use set -q var[1]. Why doesn't set -Ux (exported universal variables) seem to work?A global variable of the same name already exists.Environment variables such as EDITOR or TZ can be set universally using set -Ux. However, if there is an environment variable already set before fish starts (such as by login scripts or system administrators), it is imported into fish as a global variable. The variable scopes are searched from the "inside out", which means that local variables are checked first, followed by global variables, and finally universal variables. This means that the global value takes precedence over the universal value. To avoid this problem, consider changing the setting which fish inherits. If this is not possible, add a statement to your configuration file (usually ~/.config/fish/config.fish): set -gx EDITOR vim How do I run a command every login? What's fish's equivalent to .bashrc or .profile?Edit the file ~/.config/fish/config.fish [1], creating it if it does not exist (Note the leading period).
How do I set my prompt?The prompt is the output of the fish_prompt function. Put it in ~/.config/fish/functions/fish_prompt.fish. For example, a simple prompt is:function fish_prompt set_color $fish_color_cwd echo -n (prompt_pwd) set_color normal echo -n ' > ' end You can also use the Web configuration tool, fish_config, to preview and choose from a gallery of sample prompts. If you want to modify your existing prompt, you can use funced and funcsave like: >_ funced fish_prompt # This opens up your editor (set in $EDITOR). # Modify the function, # save the file and repeat to your liking. # Once you are happy with it: >_ funcsave fish_prompt This also applies to fish_right_prompt and fish_mode_prompt. Why does my prompt show a [I]?That's the fish_mode_prompt. It is displayed by default when you've activated vi mode using fish_vi_key_bindings.If you haven't activated vi mode on purpose, you might have installed a third-party theme that does it. If you want to change or disable this display, modify the fish_mode_prompt function, for instance via funced. How do I customize my syntax highlighting colors?Use the web configuration tool, fish_config, or alter the fish_color family of environment variables.How do I change the greeting message?Change the value of the variable fish_greeting or create a fish_greeting function. For example, to remove the greeting use:set -U fish_greeting Or if you prefer not to use a universal variable, use: set -g fish_greeting in config.fish. I'm seeing weird output before each prompt when using screen. What's wrong?Quick answer:Run the following command in fish: function fish_title; end; funcsave fish_title Problem solved! The long answer: Fish is trying to set the titlebar message of your terminal. While screen itself supports this feature, your terminal does not. Unfortunately, when the underlying terminal doesn't support setting the titlebar, screen simply passes through the escape codes and text to the underlying terminal instead of ignoring them. It is impossible to detect and resolve this problem from inside fish since fish has no way of knowing what the underlying terminal type is. For now, the only way to fix this is to unset the titlebar message, as suggested above. Note that fish has a default titlebar message, which will be used if the fish_title function is undefined. So simply unsetting the fish_title function will not work. How do I run a command from history?Type some part of the command, and then hit the ↑ (up) or ↓ (down) arrow keys to navigate through history matches. Additional default key bindings include Control+P (up) and Control+N (down). See Searchable command history for more information.Why doesn't history substitution ("!$" etc.) work?Because history substitution is an awkward interface that was invented before interactive line editing was even possible. Instead of adding this pseudo-syntax, fish opts for nice history searching and recall features. Switching requires a small change of habits: if you want to modify an old line/word, first recall it, then edit.As a special case, most of the time history substitution is used as sudo !!. In that case just press Alt+S, and it will recall your last commandline with sudo prefixed (or toggle a sudo prefix on the current commandline if there is anything). In general, fish's history recall works like this:
See documentation for more details about line editing in fish. How do I run a subcommand? The backtick doesn't work!fish uses parentheses for subcommands. For example:for i in (ls) echo $i end My command (pkg-config) gives its output as a single long string?Unlike other shells, fish splits command substitutions only on newlines, not spaces or tabs or the characters in $IFS.That means if you run echo x(printf '%s ' a b c)x It will print xa b c x, because the "a b c " is used in one piece. But if you do echo x(printf '%s\n' a b c)x it will print xax xbx xcx. In the overwhelming majority of cases, splitting on spaces is unwanted, so this is an improvement. However sometimes, especially with pkg-config and related tools, splitting on spaces is needed. In these cases use string split -n " " like: g++ example_01.cpp (pkg-config --cflags --libs gtk+-2.0 | string split -n " ") The -n is so empty elements are removed like POSIX shells would do. How do I get the exit status of a command?Use the $status variable. This replaces the $? variable used in some other shells.somecommand if test $status -eq 7 echo "That's my lucky number!" end If you are just interested in success or failure, you can run the command directly as the if-condition: if somecommand echo "Command succeeded" else echo "Command failed" end Or if you just want to do one command in case the first succeeded or failed, use and or or: somecommand or someothercommand See the documentation for test and if for more information. My command prints No matches for wildcard but works in bashIn short: quote or escape the wildcard:scp user@ip:/dir/"string-*" When fish sees an unquoted *, it performs wildcard expansion. That means it tries to match filenames to the given string. If the wildcard doesn't match any files, fish prints an error instead of running the command: > echo *this*does*not*exist fish: No matches for wildcard '*this*does*not*exist'. See `help expand`. echo *this*does*not*exist 2>| xsel --clipboard ^ Now, bash also tries to match files in this case, but when it doesn't find a match, it passes along the literal wildcard string instead. That means that commands like the above scp user@ip:/dir/string-* or apt install postgres-* appear to work, because most of the time the string doesn't match and so it passes along the string-*, which is then interpreted by the receiving program. But it also means that these commands can stop working at any moment once a matching file is encountered (because it has been created or the command is executed in a different working directory), and to deal with that bash needs workarounds like for f in ./*.mpg; do # We need to test if the file really exists because # the wildcard might have failed to match. test -f "$f" || continue mympgviewer "$f" done (from http://mywiki.wooledge.org/BashFAQ/004) For these reasons, fish does not do this, and instead expects asterisks to be quoted or escaped if they aren't supposed to be expanded. This is similar to bash's "failglob" option. I accidentally entered a directory path and fish changed directory. What happened?If fish is unable to locate a command with a given name, and it starts with ., / or ~, fish will test if a directory of that name exists. If it does, it is implicitly assumed that you want to change working directory. For example, the fastest way to switch to your home directory is to simply press ~ and enter.How can I use - as a shortcut for cd -?In fish versions prior to 2.5.0 it was possible to create a function named - that would do cd -. Changes in the 2.5.0 release included several bug fixes that enforce the rule that a bare hyphen is not a valid function (or variable) name. However, you can achieve the same effect via an abbreviation:abbr -a -- - 'cd -' The open command doesn't work.The open command uses the MIME type database and the .desktop files used by Gnome and KDE to identify filetypes and default actions. If at least one of these environments is installed, but the open command is not working, this probably means that the relevant files are installed in a non-standard location. Consider asking for more help.Why won't SSH/SCP/rsync connect properly when fish is my login shell?This problem may show up as messages like "Received message too long", "open terminal failed: not a terminal", "Bad packet length", or "Connection refused" with strange output in ssh_exchange_identification messages in the debug log.This usually happens because fish reads the user configuration file (~/.config/fish/config.fish) always, whether it's in an interactive or login or non-interactive or non-login shell. This simplifies matters, but it also means when config.fish generates output, it will do that even in non-interactive shells like the one ssh/scp/rsync start when they connect. Anything in config.fish that produces output should be guarded with status is-interactive (or status is-login if you prefer): if status is-interactive ... end The same applies for example when you start tmux in config.fish without guards, which will cause a message like sessions should be nested with care, unset $TMUX to force. I'm getting weird graphical glitches (a staircase effect, ghost characters, cursor in the wrong position,...)?In a terminal, the application running inside it and the terminal itself need to agree on the width of characters in order to handle cursor movement.This is more important to fish than other shells because features like syntax highlighting and autosuggestions are implemented by moving the cursor. Sometimes, there is disagreement on the width. There are numerous causes and fixes for this:
This also means that a few things are unsupportable:
How do I make fish my default shell?If you installed fish manually (e.g. by compiling it, not by using a package manager), you first need to add fish to the list of shells by executing the following command (assuming you installed fish in /usr/local):echo /usr/local/bin/fish | sudo tee -a /etc/shells If you installed a prepackaged version of fish, the package manager should have already done this for you. In order to change your default shell, type: chsh -s /usr/local/bin/fish You may need to adjust the above path to e.g. /usr/bin/fish. Use the command which fish if you are unsure of where fish is installed. Unfortunately, there is no way to make the changes take effect at once. You will need to log out and back in again. Uninstalling fishIf you want to uninstall fish, first make sure fish is not set as your shell. Run chsh -s /bin/bash if you are not sure.If you installed it with a package manager, just use that package manager's uninstall function. If you built fish yourself, assuming you installed it to /usr/local, do this: rm -Rf /usr/local/etc/fish /usr/local/share/fish ~/.config/fish rm /usr/local/share/man/man1/fish*.1 cd /usr/local/bin rm -f fish fish_indent Where can I find extra tools for fish?The fish user community extends fish in unique and useful ways via scripts that aren't always appropriate for bundling with the fish package. Typically because they solve a niche problem unlikely to appeal to a broad audience. You can find those extensions, including prompts, themes and useful functions, in various third-party repositories. These include:
This is not an exhaustive list and the fish project has no opinion regarding the merits of the repositories listed above or the scripts found therein. Interactive useFish prides itself on being really nice to use interactively. That's down to a few features we'll explain in the next few sections.Fish is used by giving commands in the fish language, see The Fish Language for information on that. HelpFish has an extensive help system. Use the help command to obtain help on a specific subject or command. For instance, writing help syntax displays the syntax section of this documentation.Fish also has man pages for its commands, and translates the help pages to man pages. For example, man set will show the documentation for set as a man page. Help on a specific builtin can also be obtained with the -h parameter. For instance, to obtain help on the fg builtin, either type fg -h or help fg. This page can be viewed via help index (or just help) or man fish-doc. The tutorial can be viewed with help tutorial or man fish-tutorial. Autosuggestionsfish suggests commands as you type, based on command history, completions, and valid file paths. As you type commands, you will see a suggestion offered after the cursor, in a muted gray color (which can be changed with the fish_color_autosuggestion variable).To accept the autosuggestion (replacing the command line contents), press → or Control+F. To accept the first suggested word, press Alt+→ or Alt+F. If the autosuggestion is not what you want, just ignore it: it won't execute unless you accept it. Autosuggestions are a powerful way to quickly summon frequently entered commands, by typing the first few characters. They are also an efficient technique for navigating through directory hierarchies. Tab CompletionTab completion is a time saving feature of any modern shell. When you type Tab, fish tries to guess the rest of the word under the cursor. If it finds just one possibility, it inserts it. If it finds more, it inserts the longest unambiguous part and then opens a menu (the "pager") that you can navigate to find what you're looking for.The pager can be navigated with the arrow keys, Page Up / Page Down, Tab or Shift+Tab. Pressing Control+S (the pager-toggle-search binding - / in vi-mode) opens up a search menu that you can use to filter the list. Fish provides some general purpose completions:
It also provides a large number of program specific scripted completions. Most of these completions are simple options like the -l option for ls, but some are more advanced. For example:
You can also write your own completions or install some you got from someone else. For that, see Writing your own completions. Syntax highlightingFish interprets the command line as it is typed and uses syntax highlighting to provide feedback. The most important feedback is the detection of potential errors. By default, errors are marked red.Detected errors include:
When the cursor is over a parenthesis or a quote, fish also highlights its matching quote or parenthesis. To customize the syntax highlighting, you can set the environment variables listed in the Variables for changing highlighting colors section. Syntax highlighting variablesThe colors used by fish for syntax highlighting can be configured by changing the values of a various variables. The value of these variables can be one of the colors accepted by the set_color command. The modifier switches accepted by set_color like --bold, --dim, --italics, --reverse and --underline are also accepted.Example: to make errors highlighted and red, use: set fish_color_error red --bold The following variables are available to change the highlighting colors in fish:
If a variable isn't set, fish usually tries $fish_color_normal, except for $fish_color_keyword, where it tries $fish_color_command first. Pager color variablesfish will sometimes present a list of choices in a table, called the pager.Example: to set the background of each pager row, use: set fish_pager_color_background --background=white To have black text on alternating white and gray backgrounds: set fish_pager_color_prefix black set fish_pager_color_completion black set fish_pager_color_description black set fish_pager_color_background --background=white set fish_pager_color_secondary_background --background=brwhite Variables affecting the pager colors:
When the secondary or selected variables aren't set, the normal variables are used, except for $fish_pager_color_selected_background, where the background of $fish_color_search_match is tried first. AbbreviationsTo avoid needless typing, a frequently-run command like git checkout can be abbreviated to gco using the abbr command.abbr -a gco git checkout After entering gco and pressing Space or Enter, the full text git checkout will appear in the command line. This is an alternative to aliases, and has the advantage that you see the actual command before using it, and the actual command will be stored in history. Programmable titleWhen using most virtual terminals, it is possible to set the message displayed in the titlebar of the terminal window. This can be done automatically in fish by defining the fish_title function. The fish_title function is executed before and after a new command is executed or put into the foreground and the output is used as a titlebar message. The status current-command builtin will always return the name of the job to be put into the foreground (or fish if control is returning to the shell) when the fish_prompt <cmd-fish_prompt> function is called. The first argument to fish_title will contain the most recently executed foreground command as a string, starting with fish 2.2.Examples: The default fish title is: function fish_title echo (status current-command) ' ' pwd end To show the last command in the title: function fish_title echo $argv[1] end Programmable promptWhen fish waits for input, it will display a prompt by evaluating the fish_prompt and fish_right_prompt functions. The output of the former is displayed on the left and the latter's output on the right side of the terminal. The output of fish_mode_prompt will be prepended on the left, though the default function only does this when in vi-mode.Configurable greetingIf a function named fish_greeting exists, it will be run when entering interactive mode. Otherwise, if an environment variable named fish_greeting exists, it will be printed.Private modeIf $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 test -n "$fish_private_mode" ...) if you would like to respect the user's wish for privacy and alter the behavior of your own fish scripts. Command line editorThe fish editor features copy and paste, a searchable history and many editor functions that can be bound to special keyboard shortcuts.Like bash and other shells, fish includes two sets of keyboard shortcuts (or key bindings): one inspired by the Emacs text editor, and one by the Vi text editor. The default editing mode is Emacs. You can switch to Vi mode by running fish_vi_key_bindings and switch back with fish_default_key_bindings. You can also make your own key bindings by creating a function and setting the fish_key_bindings variable to its name. For example: function fish_hybrid_key_bindings --description \ "Vi-style bindings that inherit emacs-style bindings in all modes" for mode in default insert visual fish_default_key_bindings -M $mode end fish_vi_key_bindings --no-erase end set -g fish_key_bindings fish_hybrid_key_bindings While the key bindings included with fish include many of the shortcuts popular from the respective text editors, they are not a complete implementation. They include a shortcut to open the current command line in your preferred editor (Alt+E by default) if you need the full power of your editor. Shared bindingsSome bindings are common across Emacs and Vi mode, because they aren't text editing bindings, or because what Vi/Vim does for a particular key doesn't make sense for a shell.
Emacs mode commands
You can change these key bindings using the bind builtin. Vi mode commandsVi mode allows for the use of Vi-like commands at the prompt. Initially, insert mode is active. Escape enters command mode. The commands available in command, insert and visual mode are described below. Vi mode shares some bindings with Emacs mode.It is also possible to add all emacs-mode bindings to vi-mode by using something like: function fish_user_key_bindings # Execute this once per mode that emacs bindings should be used in fish_default_key_bindings -M insert # Then execute the vi-bindings so they take precedence when there's a conflict. # Without --no-erase fish_vi_key_bindings will default to # resetting all bindings. # The argument specifies the initial mode (insert, "default" or visual). fish_vi_key_bindings --no-erase insert end When in vi-mode, the fish_mode_prompt function will display a mode indicator to the left of the prompt. To disable this feature, override it with an empty function. To display the mode elsewhere (like in your right prompt), use the output of the fish_default_mode_prompt function. When a binding switches the mode, it will repaint the mode-prompt if it exists, and the rest of the prompt only if it doesn't. So if you want a mode-indicator in your fish_prompt, you need to erase fish_mode_prompt e.g. by adding an empty file at ~/.config/fish/functions/fish_mode_prompt.fish. (Bindings that change the mode are supposed to call the repaint-mode bind function, see bind) The fish_vi_cursor function will be used to change the cursor's shape depending on the mode in supported terminals. The following snippet can be used to manually configure cursors after enabling vi-mode: # Emulates vim's cursor shape behavior # Set the normal and visual mode cursors to a block set fish_cursor_default block # Set the insert mode cursor to a line set fish_cursor_insert line # Set the replace mode cursor to an underscore set fish_cursor_replace_one underscore # The following variable can be used to configure cursor shape in # visual mode, but due to fish_cursor_default, is redundant here set fish_cursor_visual block Additionally, blink can be added after each of the cursor shape parameters to set a blinking cursor in the specified shape. If the cursor shape does not appear to be changing after setting the above variables, it's likely your terminal emulator does not support the capabilities necessary to do this. It may also be the case, however, that fish_vi_cursor has not detected your terminal's features correctly (for example, if you are using tmux). If this is the case, you can force fish_vi_cursor to set the cursor shape by setting $fish_vi_force_cursor in config.fish. You'll have to restart fish for any changes to take effect. If cursor shape setting remains broken after this, it's almost certainly an issue with your terminal emulator, and not fish. Command modeCommand mode is also known as normal mode.
Insert mode
Visual mode
Custom bindingsIn addition to the standard bindings listed here, you can also define your own with bind:# Just clear the commandline on control-c bind \cc 'commandline -r ""' Put bind statements into config.fish or a function called fish_user_key_bindings. The key sequence (the \cc) here depends on your setup, in particular the terminal. To find out what the terminal sends use fish_key_reader: > fish_key_reader # pressing control-c Press a key: hex: 3 char: \cC Press [ctrl-C] again to exit bind \cC 'do something' > fish_key_reader # pressing the right-arrow Press a key: hex: 1B char: \c[ (or \e) ( 0.077 ms) hex: 5B char: [ ( 0.037 ms) hex: 43 char: C bind \e\[C 'do something' Note that some key combinations are indistinguishable or unbindable. For instance control-i is the same as the tab key. This is a terminal limitation that fish can't do anything about. Also, Escape is the same thing as Alt in a terminal. To distinguish between pressing Escape and then another key, and pressing Alt and that key (or an escape sequence the key sends), fish waits for a certain time after seeing an escape character. This is configurable via the fish_escape_delay_ms variable. If you want to be able to press Escape and then a character and have it count as Alt+that character, set it to a higher value, e.g.: set -g fish_escape_delay_ms 100 Copy and paste (Kill Ring)Fish uses an Emacs-style kill ring for copy and paste functionality. For example, use Control+K (kill-line) to cut from the current cursor position to the end of the line. The string that is cut (a.k.a. killed in emacs-ese) is inserted into a list of kills, called the kill ring. To paste the latest value from the kill ring (emacs calls this "yanking") use Control+Y (the yank input function). After pasting, use Alt+Y (yank-pop) to rotate to the previous kill.Copy and paste from outside are also supported, both via the Control+X / Control+V bindings (the fish_clipboard_copy and fish_clipboard_paste functions [1]) and via the terminal's paste function, for which fish enables "Bracketed Paste Mode", so it can tell a paste from manually entered text. In addition, when pasting inside single quotes, pasted single quotes and backslashes are automatically escaped so that the result can be used as a single token simply by closing the quote after. Kill ring entries are stored in fish_killring variable.
Multiline editingThe fish commandline editor can be used to work on commands that are several lines long. There are three ways to make a command span more than a single line:
The fish commandline editor works exactly the same in single line mode and in multiline mode. To move between lines use the left and right arrow keys and other such keyboard shortcuts. Searchable command historyAfter a command has been executed, it is remembered in the history list. Any duplicate history items are automatically removed. By pressing the up and down keys, you can search forwards and backwards in the history. If the current command line is not empty when starting a history search, only the commands containing the string entered into the command line are shown.By pressing Alt+↑ and Alt+↓, a history search is also performed, but instead of searching for a complete commandline, each commandline is broken into separate elements just like it would be before execution, and the history is searched for an element matching that under the cursor. History searches are case-insensitive unless the search string contains an uppercase character. You can stop a search to edit your search string by pressing Esc or Page Down. Prefixing the commandline with a space will prevent the entire line from being stored in the history. The command history is stored in the file ~/.local/share/fish/fish_history (or $XDG_DATA_HOME/fish/fish_history if that variable is set) by default. However, you can set the fish_history environment variable to change the name of the history session (resulting in a <session>_history file); both before starting the shell and while the shell is running. See the history command for other manipulations. Examples: To search for previous entries containing the word 'make', type make in the console and press the up key. If the commandline reads cd m, place the cursor over the m character and press Alt+↑ to search for previously typed words containing 'm'. Navigating directoriesThe current working directory can be displayed with the pwd command, or the $PWD special variable.Directory historyFish automatically keeps a trail of the recent visited directories with cd by storing this history in the dirprev and dirnext variables.Several commands are provided to interact with this directory history:
Directory stackAnother set of commands, usually also available in other shells like bash, deal with the directory stack. Stack handling is not automatic and needs explicit calls of the following commands:
The fish languageThis document is a comprehensive overview of fish's scripting language.For interactive features see Interactive use. Syntax overviewShells like fish are used by giving them commands. A command is executed by writing the name of the command followed by any arguments. For example:echo hello world This calls the echo command. echo writes its arguments to the screen. In this example the output is hello world. Everything in fish is done with commands. There are commands for repeating other commands, commands for assigning variables, commands for treating a group of commands as a single command, etc. All of these commands follow the same basic syntax. To learn more about the echo command, read its manual page by typing man echo. man is a command for displaying a manual page on a given topic. It takes the name of the manual page to display as an argument. There are manual pages for almost every command. There are also manual pages for many other things, such as system libraries and important files. Every program on your computer can be used as a command in fish. If the program file is located in one of the PATH directories, you can just type the name of the program to use it. Otherwise the whole filename, including the directory (like /home/me/code/checkers/checkers or ../checkers) is required. Here is a list of some useful commands:
Commands and arguments are separated by the space character ' '. Every command ends with either a newline (by pressing the return key) or a semicolon ;. Multiple commands can be written on the same line by separating them with semicolons. A switch is a very common special type of argument. Switches almost always start with one or more hyphens - and alter the way a command operates. For example, the ls command usually lists the names of all files and directories in the current working directory. By using the -l switch, the behavior of ls is changed to not only display the filename, but also the size, permissions, owner, and modification time of each file. Switches differ between commands and are usually documented on a command's manual page. There are some switches, however, that are common to most commands. For example, --help will usually display a help text, --version will usually display the command version, and -i will often turn on interactive prompting before taking action. TerminologyHere we define some of the terms used on this page and throughout the rest of the fish documentation:
QuotesSometimes features like parameter expansion and character escapes get in the way. When that happens, you can use quotes, either single (') or double ("). Between single quotes, fish performs no expansions. Between double quotes, fish only performs variable expansion. No other kind of expansion (including brace expansion or parameter expansion) is performed, and escape sequences (for example, \n) are ignored. Within quotes, whitespace is not used to separate arguments, allowing quoted arguments to contain spaces.The only meaningful escape sequences in single quotes are \', which escapes a single quote and \\, which escapes the backslash symbol. The only meaningful escapes in double quotes are \", which escapes a double quote, \$, which escapes a dollar character, \ followed by a newline, which deletes the backslash and the newline, and \\, which escapes the backslash symbol. Single quotes have no special meaning within double quotes and vice versa. Example: rm "cumbersome filename.txt" removes the file cumbersome filename.txt, while rm cumbersome filename.txt removes two files, cumbersome and filename.txt. Another example: grep 'enabled)$' foo.txt searches for lines ending in enabled) in foo.txt (the $ is special to grep: it matches the end of the line). Escaping CharactersSome characters cannot be written directly on the command line. For these characters, so-called escape sequences are provided. These are:
Some characters have special meaning to the shell. For example, an apostrophe ' disables expansion (see Quotes). To tell the shell to treat these characters literally, escape them with a backslash. For example, the command: echo \'hello world\' outputs 'hello world' (including the apostrophes), while the command: echo 'hello world' outputs hello world (without the apostrophes). In the former case the shell treats the apostrophes as literal ' characters, while in the latter case it treats them as special expansion modifiers. The special characters and their escape sequences are:
Input/Output RedirectionMost programs use three input/output (I/O) streams:
Each stream has a number called the file descriptor (FD): 0 for stdin, 1 for stdout, and 2 for stderr. The destination of a stream can be changed using something called redirection. For example, echo hello > output.txt, redirects the standard output of the echo command to a text file.
DESTINATION can be one of the following:
As a convenience, the redirection &> can be used to direct both stdout and stderr to the same destination. For example, echo hello &> all_output.txt redirects both stdout and stderr to the file all_output.txt. This is equivalent to echo hello > all_output.txt 2>&1. Any arbitrary file descriptor can used in a redirection by prefixing the redirection with the FD number.
For example, echo hello 2> output.stderr writes the standard error (file descriptor 2) to output.stderr. It is an error to redirect a builtin, function, or block to a file descriptor above 2. However this is supported for external commands.
PipingAnother way to redirect streams is a pipe. A pipe connects streams with each other. Usually the standard output of one command is connected with the standard input of another. This is done by separating commands with the pipe character |. For example:cat foo.txt | head The command cat foo.txt sends the contents of foo.txt to stdout. This output is provided as input for the head program, which prints the first 10 lines of its input. It is possible to pipe a different output file descriptor by prepending its FD number and the output redirect symbol to the pipe. For example: make fish 2>| less will attempt to build fish, and any errors will be shown using the less pager. [2] As a convenience, the pipe &| redirects both stdout and stderr to the same process. This is different from bash, which uses |&.
Job controlWhen you start a job in fish, fish itself will pause, and give control of the terminal to the program just started. Sometimes, you want to continue using the commandline, and have the job run in the background. To create a background job, append an & (ampersand) to your command. This will tell fish to run the job in the background. Background jobs are very useful when running programs that have a graphical user interface.Example: emacs & will start the emacs text editor in the background. fg can be used to bring it into the foreground again when needed. Most programs allow you to suspend the program's execution and return control to fish by pressing Control+Z (also referred to as ^Z). Once back at the fish commandline, you can start other programs and do anything you want. If you then want you can go back to the suspended command by using the fg (foreground) command. If you instead want to put a suspended job into the background, use the bg command. To get a listing of all currently started jobs, use the jobs command. These listed jobs can be removed with the disown command. At the moment, functions cannot be started in the background. Functions that are stopped and then restarted in the background using the bg command will not execute correctly. FunctionsFunctions are programs written in the fish syntax. They group together various commands and their arguments using a single name.For example, here's a simple function to list directories: function ll ls -l $argv end The first line tells fish to define a function by the name of ll, so it can be used by simply writing ll on the commandline. The second line tells fish that the command ls -l $argv should be called when ll is invoked. $argv is a list variable, which always contains all arguments sent to the function. In the example above, these are simply passed on to the ls command. The end on the third line ends the definition. Calling this as ll /tmp/ will end up running ls -l /tmp/, which will list the contents of /tmp. This is a kind of function known as a wrapper or "alias". Fish's prompt is also defined in a function, called fish_prompt. It is run when the prompt is about to be displayed and its output forms the prompt: function fish_prompt # A simple prompt. Displays the current directory # (which fish stores in the $PWD variable) # and then a user symbol - a '►' for a normal user and a '#' for root. set -l user_char '►' if fish_is_root_user set user_char '#' end echo (set_color yellow)$PWD (set_color purple)$user_char end To edit a function, you can use funced, and to save a function funcsave. This will store it in a function file that fish will autoload when needed. The functions builtin can show a function's current definition (and type will also do if given a function). For more information on functions, see the documentation for the function builtin. Defining aliasesOne of the most common uses for functions is to slightly alter the behavior of an already existing command. For example, one might want to redefine the ls command to display colors. The switch for turning on colors on GNU systems is --color=auto. An alias, or wrapper, around ls might look like this:function ls command ls --color=auto $argv end There are a few important things that need to be noted about aliases:
To easily create a function of this form, you can use the alias command. Unlike other shells, this just makes functions - fish has no separate concept of an "alias", we just use the word for a function wrapper like this. For an alternative, try abbreviations. These are words that are expanded while you type, instead of being actual functions inside the shell. Autoloading functionsFunctions can be defined on the commandline or in a configuration file, but they can also be automatically loaded. This has some advantages:
When fish needs to load a function, it searches through any directories in the list variable $fish_function_path for a file with a name consisting of the name of the function plus the suffix .fish and loads the first it finds. For example if you try to execute something called banana, fish will go through all directories in $fish_function_path looking for a file called banana.fish and load the first one it finds. By default $fish_function_path contains the following:
If you are unsure, your functions probably belong in ~/.config/fish/functions. As we've explained, autoload files are loaded by name, so, while you can put multiple functions into one file, the file will only be loaded automatically once you try to execute the one that shares the name. Autoloading also won't work for event handlers, since fish cannot know that a function is supposed to be executed when an event occurs when it hasn't yet loaded the function. See the event handlers section for more information. If a file of the right name doesn't define the function, fish will not read other autoload files, instead it will go on to try builtins and finally commands. This allows masking a function defined later in $fish_function_path, e.g. if your administrator has put something into /etc/fish/functions that you want to skip. If you are developing another program and want to install fish functions for it, install them to the "vendor" functions directory. As this path varies from system to system, you can use pkgconfig to discover it with the output of pkg-config --variable functionsdir fish. Your installation system should support a custom path to override the pkgconfig path, as other distributors may need to alter it easily. CommentsAnything after a # until the end of the line is a comment. That means it's purely for the reader's benefit, fish ignores it.This is useful to explain what and why you are doing something: function ls # The function is called ls, # so we have to explicitly call `command ls` to avoid calling ourselves. command ls --color=auto $argv end There are no multiline comments. If you want to make a comment span multiple lines, simply start each line with a #. Comments can also appear after a line like so: set -gx EDITOR emacs # I don't like vim. ConditionsFish has some builtins that let you execute commands only if a specific criterion is met: if, switch, and and or, and also the familiar &&/|| syntax.The switch command is used to execute one of possibly many blocks of commands depending on the value of a string. See the documentation for switch for more information. The other conditionals use the exit status of a command to decide if a command or a block of commands should be executed. Unlike programming languages you might know, if doesn't take a condition, it takes a command. If that command returned a successful exit status (that's 0), the if branch is taken, otherwise the else branch. Some examples: # Just see if the file contains the string "fish" anywhere. # This executes the `grep` command, which searches for a string, # and if it finds it returns a status of 0. # The `-q` switch stops it from printing any matches. if grep -q fish myanimals echo "You have fish!" else echo "You don't have fish!" end # $XDG_CONFIG_HOME is a standard place to store configuration. # If it's not set applications should use ~/.config. set -q XDG_CONFIG_HOME; and set -l configdir $XDG_CONFIG_HOME or set -l configdir ~/.config For more, see the documentation for the builtins or the Conditionals section of the tutorial. Loops and blocksLike most programming language, fish also has the familiar while and for loops.while works like a repeated if: while true echo Still running sleep 1 end will print "Still running" once a second. You can abort it with ctrl-c. for loops work like in other shells, which is more like python's for-loops than e.g. C's: for file in * echo file: $file end will print each file in the current directory. The part after the in is just a list of arguments, so you can use any expansions there: set moreanimals bird fox for animal in {cat,}fish dog $moreanimals echo I like the $animal end If you need a list of numbers, you can use the seq command to create one: for i in (seq 1 5) echo $i end break is available to break out of a loop, and continue to jump to the next iteration. Input and output redirections (including pipes) can also be applied to loops: while read -l line echo line: $line end < file In addition there's a begin block that just groups commands together so you can redirect to a block or use a new variable scope without any repetition: begin set -l foo bar # this variable will only be available in this block! end Parameter expansionWhen fish is given a commandline, it expands the parameters before sending them to the command. There are multiple different kinds of expansions:
Parameter expansion is limited to 524288 items. There is a limit to how many arguments the operating system allows for any command, and 524288 is far above it. This is a measure to stop the shell from hanging doing useless computation. Wildcards ("Globbing")When a parameter includes an unquoted * star (or "asterisk") or a ? question mark, fish uses it as a wildcard to match files.
Other shells, such as zsh, have a much richer glob syntax, like **(.) to only match regular files. Fish does not. Instead of reinventing the wheel, use programs like find to look for files. For example: function ff --description 'Like ** but only returns plain files.' # This also ignores .git directories. find . \( -name .git -type d -prune \) -o -type f | \ sed -n -e '/^\.\/\.git$/n' -e 's/^\.\///p' end You would then use it in place of ** like this, my_prog (ff), to pass only regular files in or below $PWD to my_prog. [3] Wildcard matches are sorted case insensitively. When sorting matches containing numbers, they are naturally sorted, so that the strings '1' '5' and '12' would be sorted like 1, 5, 12. Hidden files (where the name begins with a dot) are not considered when wildcarding unless the wildcard string has a dot in that place. Examples:
For most commands, if any wildcard fails to expand, the command is not executed, $status is set to nonzero, and a warning is printed. This behavior is like what bash does with shopt -s failglob. There are exactly 4 exceptions, namely set, overriding variables in overrides, count and for. Their globs will instead expand to zero arguments (so the command won't see them at all), like with shopt -s nullglob in bash. Examples: # List the .foo files, or warns if there aren't any. ls *.foo # List the .foo files, if any. set foos *.foo if count $foos >/dev/null ls $foos end
Variable expansionOne of the most important expansions in fish is the "variable expansion". This is the replacing of a dollar sign ($) followed by a variable name with the _value_ of that variable. For more on shell variables, read the Shell variables section.In the simplest case, this is just something like: echo $HOME which will replace $HOME with the home directory of the current user, and pass it to echo, which will then print it. Some variables like $HOME are already set because fish sets them by default or because fish's parent process passed them to fish when it started it. You can define your own variables by setting them with set: set my_directory /home/cooluser/mystuff ls $my_directory # shows the contents of /home/cooluser/mystuff For more on how setting variables works, see Shell variables and the following sections. Sometimes a variable has no value because it is undefined or empty, and it expands to nothing: echo $nonexistentvariable # Prints no output. To separate a variable name from text you can encase the variable within double-quotes or braces: set WORD cat echo The plural of $WORD is "$WORD"s # Prints "The plural of cat is cats" because $WORD is set to "cat". echo The plural of $WORD is {$WORD}s # ditto Without the quotes or braces, fish will try to expand a variable called $WORDs, which may not exist. The latter syntax {$WORD} is a special case of brace expansion. If $WORD here is undefined or an empty list, the "s" is not printed. However, it is printed if $WORD is the empty string (like after set WORD ""). Unlike all the other expansions, variable expansion also happens in double quoted strings. Inside double quotes ("these"), variables will always expand to exactly one argument. If they are empty or undefined, it will result in an empty string. If they have one element, they'll expand to that element. If they have more than that, the elements will be joined with spaces, unless the variable is a path variable - in that case it will use a colon (:) instead [4]. Outside of double quotes, variables will expand to as many arguments as they have elements. That means an empty list will expand to nothing, a variable with one element will expand to that element, and a variable with multiple elements will expand to each of those elements separately. If a variable expands to nothing, it will cancel out any other strings attached to it. See the cartesian product section for more information. The $ symbol can also be used multiple times, as a kind of "dereference" operator (the * in C or C++), like in the following code: set foo a b c set a 10; set b 20; set c 30 for i in (seq (count $$foo)) echo $$foo[$i] end # Output is: # 10 # 20 # 30 $$foo[$i] is "the value of the variable named by $foo[$i]. When using this feature together with list brackets, the brackets will be used from the inside out. $$foo[5] will use the fifth element of $foo as a variable name, instead of giving the fifth element of all the variables $foo refers to. That would instead be expressed as $$foo[1][5] (take the first element of $foo, use it as a variable name, then give the fifth element of that).
Command substitutionThe output of a command (or an entire pipeline) can be used as the arguments to another command.When you write a command in parenthesis like outercommand (innercommand), the innercommand will be executed first. Its output will be taken and each line given as a separate argument to outercommand, which will then be executed. [5] If the output is piped to string split or string split0 as the last step, those splits are used as they appear instead of splitting lines. The exit status of the last run command substitution is available in the status variable if the substitution happens in the context of a set command (so if set -l (something) checks if something returned true). Only part of the output can be used, see index range expansion for details. Fish has a default limit of 100 MiB on the data it will read in a command sustitution. If that limit is reached the command (all of it, not just the command substitution - the outer command won't be executed at all) fails and $status is set to 122. This is so command substitutions can't cause the system to go out of memory, because typically your operating system has a much lower limit, so reading more than that would be useless and harmful. This limit can be adjusted with the fish_read_limit variable (0 meaning no limit). This limit also affects the read command. Examples: # Outputs 'image.png'. echo (basename image.jpg .jpg).png # Convert all JPEG files in the current directory to the # PNG format using the 'convert' program. for i in *.jpg; convert $i (basename $i .jpg).png; end # Set the ``data`` variable to the contents of 'data.txt' # without splitting it into a list. begin; set -l IFS; set data (cat data.txt); end # Set ``$data`` to the contents of data, splitting on NUL-bytes. set data (cat data | string split0) Sometimes you want to pass the output of a command to another command that only accepts files. If it's just one file, you can usually just pass it via a pipe, like: grep fish myanimallist1 | wc -l but if you need multiple or the command doesn't read from standard input, "process substitution" is useful. Other shells [6] allow this via foo <(bar) <(baz), and fish uses the psub command: # Compare just the lines containing "fish" in two files: diff -u (grep fish myanimallist1 | psub) (grep fish myanimallist2 | psub) This creates a temporary file, stores the output of the command in that file and prints the filename, so it is given to the outer command.
Brace expansionCurly braces can be used to write comma-separated lists. They will be expanded with each element becoming a new parameter, with the surrounding string attached. This is useful to save on typing, and to separate a variable name from surrounding text.Examples: > echo input.{c,h,txt} input.c input.h input.txt # Move all files with the suffix '.c' or '.h' to the subdirectory src. > mv *.{c,h} src/ # Make a copy of `file` at `file.bak`. > cp file{,.bak} > set -l dogs hot cool cute "good " > echo {$dogs}dog hotdog cooldog cutedog good dog If there is no "," or variable expansion between the curly braces, they will not be expanded: # This {} isn't special > echo foo-{} foo-{} # This passes "HEAD@{2}" to git > git reset --hard HEAD@{2} > echo {{a,b}} {a} {b} # because the inner brace pair is expanded, but the outer isn't. If after expansion there is nothing between the braces, the argument will be removed (see the cartesian product section): > echo foo-{$undefinedvar} # Output is an empty line, just like a bare `echo`. If there is nothing between a brace and a comma or two commas, it's interpreted as an empty element: > echo {,,/usr}/bin /bin /bin /usr/bin To use a "," as an element, quote or escape it. Combining lists (Cartesian Product)When lists are expanded with other parts attached, they are expanded with these parts still attached. Even if two lists are attached to each other, they are expanded in all combinations. This is referred to as the cartesian product (like in mathematics), and works basically like brace expansion.Examples: # Brace expansion is the most familiar: # All elements in the brace combine with the parts outside of the braces >_ echo {good,bad}" apples" good apples bad apples # The same thing happens with variable expansion. >_ set -l a x y z >_ set -l b 1 2 3 # $a is {x,y,z}, $b is {1,2,3}, # so this is `echo {x,y,z}{1,2,3}` >_ echo $a$b x1 y1 z1 x2 y2 z2 x3 y3 z3 # Same thing if something is between the lists >_ echo $a"-"$b x-1 y-1 z-1 x-2 y-2 z-2 x-3 y-3 z-3 # Or a brace expansion and a variable >_ echo {x,y,z}$b x1 y1 z1 x2 y2 z2 x3 y3 z3 # A combined brace-variable expansion >_ echo {$b}word 1word 2word 3word # Special case: If $c has no elements, this expands to nothing >_ echo {$c}word # Output is an empty line Sometimes this may be unwanted, especially that tokens can disappear after expansion. In those cases, you should double-quote variables - echo "$c"word. This also happens after command substitution. To avoid tokens disappearing there, make the inner command return a trailing newline, or store the output in a variable and double-quote it. E.g. >_ set b 1 2 3 >_ echo (echo x)$b x1 x2 x3 >_ echo (printf '%s' '')banana # the printf prints nothing, so this is nothing times "banana", # which is nothing. >_ echo (printf '%s\n' '')banana # the printf prints a newline, # so the command substitution expands to an empty string, # so this is `''banana` banana This can be quite useful. For example, if you want to go through all the files in all the directories in $PATH, use: for file in $PATH/* Because $PATH is a list, this expands to all the files in all the directories in it. And if there are no directories in $PATH, the right answer here is to expand to no files. Index range expansionSometimes it's necessary to access only some of the elements of a list (all fish variables are lists), or some of the lines a command substitution outputs. Both are possible in fish by writing a set of indices in brackets, like:# Make $var a list of four elements set var one two three four # Print the second: echo $var[2] # prints "two" # or print the first three: echo $var[1..3] # prints "one two three" In index brackets, fish understands ranges written like a..b ('a' and 'b' being indices). They are expanded into a sequence of indices from a to b (so a a+1 a+2 ... b), going up if b is larger and going down if a is larger. Negative indices can also be used - they are taken from the end of the list, so -1 is the last element, and -2 the one before it. If an index doesn't exist the range is clamped to the next possible index. If a list has 5 elements the indices go from 1 to 5, so a range of 2..16 will only go from element 2 to element 5. If the end is negative the range always goes up, so 2..-2 will go from element 2 to 4, and 2..-16 won't go anywhere because there is no way to go from the second element to one that doesn't exist, while going up. If the start is negative the range always goes down, so -2..1 will go from element 4 to 1, and -16..2 won't go anywhere because there is no way to go from an element that doesn't exist to the second element, while going down. A missing starting index in a range defaults to 1. This is allowed if the range is the first index expression of the sequence. Similarly, a missing ending index, defaulting to -1 is allowed for the last index range in the sequence. Multiple ranges are also possible, separated with a space. Some examples: echo (seq 10)[1 2 3] # Prints: 1 2 3 # Limit the command substitution output echo (seq 10)[2..5] # Uses elements from 2 to 5 # Output is: 2 3 4 5 echo (seq 10)[7..] # Prints: 7 8 9 10 # Use overlapping ranges: echo (seq 10)[2..5 1..3] # Takes elements from 2 to 5 and then elements from 1 to 3 # Output is: 2 3 4 5 1 2 3 # Reverse output echo (seq 10)[-1..1] # Uses elements from the last output line to # the first one in reverse direction # Output is: 10 9 8 7 6 5 4 3 2 1 # The command substitution has only one line, # so these will result in empty output: echo (echo one)[2..-1] echo (echo one)[-3..1] The same works when setting or expanding variables: # Reverse path variable set PATH $PATH[-1..1] # or set PATH[-1..1] $PATH # Use only n last items of the PATH set n -3 echo $PATH[$n..-1] Variables can be used as indices for expansion of variables, like so: set index 2 set letters a b c d echo $letters[$index] # returns 'b' However using variables as indices for command substitution is currently not supported, so: echo (seq 5)[$index] # This won't work set sequence (seq 5) # It needs to be written on two lines like this. echo $sequence[$index] # returns '2' When using indirect variable expansion with multiple $ ($$name), you have to give all indices up to the variable you want to slice: > set -l list 1 2 3 4 5 > set -l name list > echo $$name[1] 1 2 3 4 5 > echo $$name[1..-1][1..3] # or $$name[1][1..3], since $name only has one element. 1 2 3 Home directory expansionThe ~ (tilde) character at the beginning of a parameter, followed by a username, is expanded into the home directory of the specified user. A lone ~, or a ~ followed by a slash, is expanded into the home directory of the process owner:ls ~/Music # lists my music directory echo ~root # prints root's home directory, probably "/root" Combining different expansionsAll of the above expansions can be combined. If several expansions result in more than one parameter, all possible combinations are created.When combining multiple parameter expansions, expansions are performed in the following order:
Expansions are performed from right to left, nested bracket expansions are performed from the inside and out. Example: If the current directory contains the files 'foo' and 'bar', the command echo a(ls){1,2,3} will output abar1 abar2 abar3 afoo1 afoo2 afoo3. Shell variablesVariables are a way to save data and pass it around. They can be used just by the shell, or they can be "exported", so that a copy of the variable is available to any external command the shell starts. An exported variable is referred to as an "environment variable".To set a variable value, use the set command. A variable name can not be empty and can contain only letters, digits, and underscores. It may begin and end with any of those characters. Example: To set the variable smurf_color to the value blue, use the command set smurf_color blue. After a variable has been set, you can use the value of a variable in the shell through variable expansion. Example: set smurf_color blue echo Smurfs are usually $smurf_color set pants_color red echo Papa smurf, who is $smurf_color, wears $pants_color pants So you set a variable with set, and use it with a $ and the name. Variable scopeThere are three kinds of variables in fish: universal, global and local variables.
Variables can be explicitly set to be universal with the -U or --universal switch, global with the -g or --global switch, or local with the -l or --local switch. The scoping rules when creating or updating a variable are:
There can be many variables with the same name, but different scopes. When you use a variable, the smallest scoped variable of that name will be used. If a local variable exists, it will be used instead of the global or universal variable of the same name. Example: There are a few possible uses for different scopes. Typically inside funcions you should use local scope: function something set -l file /path/to/my/file if not test -e "$file" set file /path/to/my/otherfile end end If you want to set something in config.fish, or set something in a function and have it available for the rest of the session, global scope is a good choice: # Don't shorten the working directory in the prompt set -g fish_prompt_pwd_dir_length 0 # Set my preferred cursor style: function setcursors set -g fish_cursor_default block set -g fish_cursor_insert line set -g fish_cursor_visual underscore end # Set my language set -gx LANG de_DE.UTF-8 If you want to set some personal customization, universal variables are nice: # Typically you'd run this interactively, fish takes care of keeping it. set -U fish_color_autosuggestion 555 Here is an example of local vs function-scoped variables: begin # This is a nice local scope where all variables will die set -l pirate 'There be treasure in them thar hills' set captain Space, the final frontier end echo $pirate # This will not output anything, since the pirate was local echo $captain # This will output the good Captain's speech since $captain had function-scope. Overriding variables for a single commandIf you want to override a variable for a single command, you can use "var=val" statements before the command:# Call git status on another directory # (can also be done via `git -C somerepo status`) GIT_DIR=somerepo git status Unlike other shells, fish will first set the variable and then perform other expansions on the line, so: set foo banana foo=gagaga echo $foo # prints gagaga, while in other shells it might print "banana" Multiple elements can be given in a brace expansion: # Call bash with a reasonable default path. PATH={/usr,}/{s,}bin bash Or with a glob: # Run vlc on all mp3 files in the current directory # If no file exists it will still be run with no arguments mp3s=*.mp3 vlc $mp3s Unlike other shells, this does not inhibit any lookup (aliases or similar). Calling a command after setting a variable override will result in the exact same command being run. This syntax is supported since fish 3.1. More on universal variablesUniversal variables are variables that are shared between all the user's fish sessions on the computer. Fish stores many of its configuration options as universal variables. This means that in order to change fish settings, all you have to do is change the variable value once, and it will be automatically updated for all sessions, and preserved across computer reboots and login/logout.To see universal variables in action, start two fish sessions side by side, and issue the following command in one of them set fish_color_cwd blue. Since fish_color_cwd is a universal variable, the color of the current working directory listing in the prompt will instantly change to blue on both terminals. Universal variables are stored in the file .config/fish/fish_variables. Do not edit this file directly, as your edits may be overwritten. Edit the variables through fish scripts or by using fish interactively instead. Do not append to universal variables in config.fish, because these variables will then get longer with each new shell instance. Instead, simply set them once at the command line. Variable scope for functionsWhen calling a function, all current local variables temporarily disappear. This shadowing of the local scope is needed since the variable namespace would become cluttered, making it very easy to accidentally overwrite variables from another function.For example: function shiver set phrase 'Shiver me timbers' end function avast set --local phrase 'Avast, mateys' # Calling the shiver function here can not # change any variables in the local scope shiver echo $phrase end avast # Outputs "Avast, mateys" Exporting variablesVariables in fish can be "exported", so they will be inherited by any commands started by fish. In particular, this is necessary for variables used to configure external commands like $LESS or $GOPATH, but also for variables that contain general system settings like $PATH or $LANGUAGE. If an external command needs to know a variable, it needs to be exported.This also applies to fish - when it starts up, it receives environment variables from its parent (usually the terminal). These typically include system configuration like $PATH and locale variables. Variables can be explicitly set to be exported with the -x or --export switch, or not exported with the -u or --unexport switch. The exporting rules when setting a variable are identical to the scoping rules for variables:
As a naming convention, exported variables are in uppercase and unexported variables are in lowercase. For example: set -gx ANDROID_HOME ~/.android # /opt/android-sdk set -gx CDPATH . ~ (test -e ~/Videos; and echo ~/Videos) set -gx EDITOR emacs -nw set -gx GOPATH ~/dev/go set -gx GTK2_RC_FILES "$XDG_CONFIG_HOME/gtk-2.0/gtkrc" set -gx LESSHISTFILE "-" Note: Exporting is not a scope, but an additional state. It typically makes sense to make exported variables global as well, but local-exported variables can be useful if you need something more specific than Overrides. They are copied to functions so the function can't alter them outside, and still available to commands. ListsFish can store a list (or an "array" if you wish) of multiple strings inside of a variable:> set mylist first second third > printf '%s\n' $mylist # prints each element on its own line first second third To access one element of a list, use the index of the element inside of square brackets, like this: echo $PATH[3] List indices start at 1 in fish, not 0 like in other languages. This is because it requires less subtracting of 1 and many common Unix tools like seq work better with it (seq 5 prints 1 to 5, not 0 to 5). An invalid index is silently ignored resulting in no value (not even an empty string, just no argument at all). If you don't use any brackets, all the elements of the list will be passed to the command as separate items. This means you can iterate over a list with for: for i in $PATH echo $i is in the path end This goes over every directory in $PATH separately and prints a line saying it is in the path. To create a variable smurf, containing the items blue and small, simply write: set smurf blue small It is also possible to set or erase individual elements of a list: # Set smurf to be a list with the elements 'blue' and 'small' set smurf blue small # Change the second element of smurf to 'evil' set smurf[2] evil # Erase the first element set -e smurf[1] # Output 'evil' echo $smurf If you specify a negative index when expanding or assigning to a list variable, the index will be taken from the end of the list. For example, the index -1 is the last element of the list: > set fruit apple orange banana > echo $fruit[-1] banana > echo $fruit[-2..-1] orange banana > echo $fruit[-1..1] # reverses the list banana orange apple As you see, you can use a range of indices, see index range expansion for details. All lists are one-dimensional and can't contain other lists, although it is possible to fake nested lists using dereferencing - see variable expansion. When a list is exported as an environment variable, it is either space or colon delimited, depending on whether it is a path variable: > set -x smurf blue small > set -x smurf_PATH forest mushroom > env | grep smurf smurf=blue small smurf_PATH=forest:mushroom Fish automatically creates lists from all environment variables whose name ends in PATH (like $PATH, $CDPATH or $MANPATH), by splitting them on colons. Other variables are not automatically split. Lists can be inspected with the count or the contains commands: count $smurf # 2 contains blue $smurf # key found, exits with status 0 > contains -i blue $smurf 1 A nice thing about lists is that they are passed to commands one element as one argument, so once you've set your list, you can just pass it: set -l grep_args -r "my string" grep $grep_args . # will run the same as `grep -r "my string"` . Unlike other shells, fish does not do "word splitting" - elements in a list stay as they are, even if they contain spaces or tabs. Argument HandlingAn important list is $argv, which contains the arguments to a function or script. For example:function myfunction echo $argv[1] echo $argv[3] end This function takes whatever arguments it gets and prints the first and third: > myfunction first second third first third > myfunction apple cucumber banana apple banana Commandline tools often get various options and flags and positional arguments, and $argv would contain all of these. A more robust approach to argument handling is argparse, which checks the defined options and puts them into various variables, leaving only the positional arguments in $argv. Here's a simple example: function mybetterfunction argparse h/help s/second -- $argv or return # exit if argparse failed because it found an option it didn't recognize - it will print an error # If -h or --help is given, we print a little help text and return if set -q _flag_help echo "mybetterfunction [-h|--help] [-s|--second] [ARGUMENTS...]" return 0 end # If -s or --second is given, we print the second argument, # not the first and third. if set -q _flag_second echo $argv[2] else echo $argv[1] echo $argv[3] end end The options will be removed from $argv, so $argv[2] is the second positional argument now: > mybetterfunction first -s second third second PATH variablesPath variables are a special kind of variable used to support colon-delimited path lists including PATH, CDPATH, MANPATH, PYTHONPATH, etc. All variables that end in "PATH" (case-sensitive) become PATH variables.PATH variables act as normal lists, except they are implicitly joined and split on colons. set MYPATH 1 2 3 echo "$MYPATH" # 1:2:3 set MYPATH "$MYPATH:4:5" echo $MYPATH # 1 2 3 4 5 echo "$MYPATH" # 1:2:3:4:5 Variables can be marked or unmarked as PATH variables via the --path and --unpath options to set. Special variablesYou can change the settings of fish by changing the values of certain variables.
Fish also provides additional information through the values of certain environment variables. Most of these variables are read-only and their value can't be changed with set.
As a convention, an uppercase name is usually used for exported variables, while lowercase variables are not exported. (CMD_DURATION is an exception for historical reasons). This rule is not enforced by fish, but it is good coding practice to use casing to distinguish between exported and unexported variables. Fish also uses some variables internally, their name usually starting with __fish. These are internal and should not typically be modified directly. The status variableWhenever a process exits, an exit status is returned to the program that started it (usually the shell). This exit status is an integer number, which tells the calling application how the execution of the command went. In general, a zero exit status means that the command executed without problem, but a non-zero exit status means there was some form of problem.Fish stores the exit status of the last process in the last job to exit in the status variable. If fish encounters a problem while executing a command, the status variable may also be set to a specific value:
If a process exits through a signal, the exit status will be 128 plus the number of the signal. The status can be negated with not (or !), which is useful in a condition. This turns a status of 0 into 1 and any non-zero status into 0. There is also $pipestatus, which is a list of all status values of processes in a pipe. One difference is that not applies to $status, but not $pipestatus, because it loses information. For example: not cat file | grep -q fish echo status is: $status pipestatus is $pipestatus Here $status reflects the status of grep, which returns 0 if it found something, negated with not (so 1 if it found something, 0 otherwise). $pipestatus reflects the status of cat (which returns non-zero for example when it couldn't find the file) and grep, without the negation. So if both cat and grep succeeded, $status would be 1 because of the not, and $pipestatus would be 0 and 0. Locale variablesThe "locale" of a program is its set of language and regional settings. In UNIX, there are a few separate variables to control separate things - LC_CTYPE defines the text encoding while LC_TIME defines the time format.The locale variables are: LANG, LC_ALL, LC_COLLATE, LC_CTYPE, LC_MESSAGES, LC_MONETARY, LC_NUMERIC and LC_TIME. These variables work as follows: LC_ALL forces all the aspects of the locale to the specified value. If LC_ALL is set, all other locale variables will be ignored (this is typically not recommended!). The other LC_ variables set the specified aspect of the locale information. LANG is a fallback value, it will be used if none of the LC_ variables are specified. The most common way to set the locale to use a command like set -gx LANG en_GB.utf8, which sets the current locale to be the English language, as used in Great Britain, using the UTF-8 character set. That way any program that requires one setting differently can easily override just that and doesn't have to resort to LC_ALL. For a list of available locales on your system, try locale -a. Because it needs to handle output that might include multibyte characters (like e.g. emojis), fish will try to set its own internal LC_CTYPE to one that is UTF8-capable even if given an effective LC_CTYPE of "C" (the default). This prevents issues with e.g. filenames given in autosuggestions even if the user started fish with LC_ALL=C. To turn this handling off, set fish_allow_singlebyte_locale to "1". Builtin commandsFish includes a number of commands in the shell directly. We call these "builtins". These include:
For a list of all builtins, use builtin -n. For a list of all builtins, functions and commands shipped with fish, see the list of commands. The documentation is also available by using the --help switch. Shell variable and function namesThe names given to variables and functions (so called "identifiers") have to follow certain rules:
Other things have other restrictions. For instance what is allowed for file names depends on your system, but at the very least they cannot contain a "/" (because that is the path separator) or NULL byte (because that is how UNIX ends strings). Future feature flagsFeature flags are how fish stages changes that might break scripts. Breaking changes are introduced as opt-in, in a few releases they become opt-out, and eventually the old behavior is removed.You can see the current list of features via status features: > status features stderr-nocaret on 3.0 ^ no longer redirects stderr qmark-noglob off 3.0 ? no longer globs regex-easyesc off 3.1 string replace -r needs fewer \\'s There are two breaking changes in fish 3.0: caret ^ no longer redirects stderr, and question mark ? is no longer a glob. There is one breaking change in fish 3.1: string replace -r does a superfluous round of escaping for the replacement, so escaping backslashes would look like string replace -ra '([ab])' '\\\\\\\$1' a. This flag removes that if turned on, so '\\\\$1' is enough. These changes are off by default. They can be enabled on a per session basis: > fish --features qmark-noglob,stderr-nocaret or opted into globally for a user: > set -U fish_features stderr-nocaret qmark-noglob Features will only be set on startup, so this variable will only take effect if it is universal or exported. You can also use the version as a group, so 3.0 is equivalent to "stderr-nocaret" and "qmark-noglob". Prefixing a feature with no- turns it off instead. Event handlersWhen defining a new function in fish, it is possible to make it into an event handler, i.e. a function that is automatically run when a specific event takes place. Events that can trigger a handler currently are:
Example: To specify a signal handler for the WINCH signal, write: function my_signal_handler --on-signal WINCH echo Got WINCH signal! end Please note that event handlers only become active when a function is loaded, which means you need to otherwise source or execute a function instead of relying on autoloading. One approach is to put it into your configuration file. For more information on how to define new event handlers, see the documentation for the function command. Debugging fish scriptsFish includes a built in debugging facility. The debugger allows you to stop execution of a script at an arbitrary point. When this happens you are presented with an interactive prompt. At this prompt you can execute any fish command (there are no debug commands as such). For example, you can check or change the value of any variables using printf and set. As another example, you can run status print-stack-trace to see how this breakpoint was reached. To resume normal execution of the script, simply type exit or Control+D.To start a debug session simply run the builtin command breakpoint at the point in a function or script where you wish to gain control. Also, the default action of the TRAP signal is to call this builtin. So a running script can be debugged by sending it the TRAP signal with the kill command. Once in the debugger, it is easy to insert new breakpoints by using the funced function to edit the definition of a function. Commandsfish ships with the following commands:_ - call fish's translationsSynopsis_ STRING... Description_ translates its arguments into the current language, if possible.It is equivalent to gettext fish STRING, meaning it can only be used to look up fish's own translations. It requires fish to be built with gettext support. If that support is disabled, or there is no translation it will simply echo the argument back. The language depends on the current locale, set with $LANG and $LC_MESSAGES. Options_ has no options.Examples> _ File Datei abbr - manage fish abbreviationsSynopsisabbr --add [SCOPE] WORD EXPANSION abbr --erase WORD... abbr --rename [SCOPE] OLD_WORD NEW_WORD abbr --show abbr --list abbr --query WORD... Descriptionabbr manages abbreviations - user-defined words that are replaced with longer phrases after they are entered.For example, a frequently-run command like git checkout can be abbreviated to gco. After entering gco and pressing Space or Enter, the full text git checkout will appear in the command line. OptionsThe following options are available:
In addition, when adding or renaming abbreviations:
See the "Internals" section for more on them. Examplesabbr -a -g gco git checkout Add a new abbreviation where gco will be replaced with git checkout global to the current shell. This abbreviation will not be automatically visible to other shells unless the same command is run in those shells (such as when executing the commands in config.fish). abbr -a -U l less Add a new abbreviation where l will be replaced with less universal so all shells. Note that you omit the -U since it is the default. abbr -r gco gch Renames an existing abbreviation from gco to gch. abbr -e gco Erase the gco abbreviation. ssh another_host abbr -s | source Import the abbreviations defined on another_host over SSH. InternalsEach abbreviation is stored in its own global or universal variable. The name consists of the prefix _fish_abbr_ followed by the WORD after being transformed by string escape style=var. The WORD cannot contain a space but all other characters are legal.Defining an abbreviation with global scope is slightly faster than universal scope (which is the default). But in general you'll only want to use the global scope when defining abbreviations in a startup script like ~/.config/fish/config.fish like this: if status --is-interactive abbr --add --global first 'echo my first abbreviation' abbr --add --global second 'echo my second abbreviation' abbr --add --global gco git checkout # etcetera end You can create abbreviations interactively and they will be visible to other fish sessions if you use the -U or --universal flag or don't explicitly specify the scope and the abbreviation isn't already defined with global scope. If you want it to be visible only to the current shell use the -g or --global flag. alias - create a functionSynopsisalias alias [OPTIONS] NAME DEFINITION alias [OPTIONS] NAME=DEFINITION Descriptionalias is a simple wrapper for the function builtin, which creates a function wrapping a command. It has similar syntax to POSIX shell alias. For other uses, it is recommended to define a function.fish marks functions that have been created by alias by including the command used to create them in the function description. You can list alias-created functions by running alias without arguments. They must be erased using functions -e.
You cannot create an alias to a function with the same name. Note that spaces need to be escaped in the call to alias just like at the command line, even inside quoted parts. The following options are available:
ExampleThe following code will create rmi, which runs rm with additional arguments on every invocation.alias rmi="rm -i" # This is equivalent to entering the following function: function rmi --wraps rm --description 'alias rmi=rm -i' rm -i $argv end # This needs to have the spaces escaped or "Chrome.app..." # will be seen as an argument to "/Applications/Google": alias chrome='/Applications/Google\ Chrome.app/Contents/MacOS/Google\ Chrome banana' See more
and - conditionally execute a commandSynopsisCOMMAND1; and COMMAND2 Descriptionand is used to execute a command if the previous command was successful (returned a status of 0).and statements may be used as part of the condition in an while or if block. and does not change the current exit status itself, but the command it runs most likely will. The exit status of the last foreground command to exit can always be accessed using the $status variable. ExampleThe following code runs the make command to build a program. If the build succeeds, make's exit status is 0, and the program is installed. If either step fails, the exit status is 1, and make clean is run, which removes the files created by the build process.make; and make install; or make clean See Also
argparse - parse options passed to a fish script or functionSynopsisargparse [OPTIONS] OPTION_SPEC... -- [ARG...] DescriptionThis command makes it easy for fish scripts and functions to handle arguments like how fish builtin commands handle their arguments. You pass arguments that define the known options, followed by a literal --, then the arguments to be parsed (which might also include a literal --). argparse then sets variables to indicate the passed options with their values, and sets $argv (and always $argv) to the remaining arguments. More on this in the usage section below.Each option specification (OPTION_SPEC) is written in the domain specific language 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 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. 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. OptionsThe following argparse options are available. They must appear before all OPTION_SPECs:
UsageTo use this command, pass the option specifications (OPTION_SPEC), then a mandatory --, and then the arguments you want to have parsed.A simple example: argparse --name=my_function 'h/help' 'n/name=' -- $argv or return If $argv is empty then there is nothing to parse and argparse returns zero to indicate success. If $argv is not empty then it is checked for flags -h, --help, -n and --name. If they are found they are removed from the arguments and local variables called _flag_OPTION are set so the script can determine which options were seen. If $argv doesn't have any errors, like a missing mandatory value for an option, then argparse exits with a status of zero. Otherwise it writes appropriate error messages to stderr and exits with a status of one. The or return means that the function returns argparse's status if it failed, so if it goes on argparse succeeded. The -- argument is required. You do not have to include any arguments after the -- but you must include the --. For example, this is acceptable: set -l argv argparse 'h/help' 'n/name' -- $argv But this is not: set -l argv argparse 'h/help' 'n/name' $argv The first -- seen is what allows the argparse command to reliably separate the option specifications and options to argparse itself (like --ignore-unknown) from the command arguments, so it is required. Option SpecificationsEach option specification consists of:
See the fish_opt command for a friendlier but more verbose way to create option specifications. If a flag is not seen when parsing the arguments then the corresponding _flag_X var(s) will not be set. Integer flagSometimes 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 argumentsAn option defined with =? can take optional arguments. Optional arguments have to be directly attached to the option they belong to.That means the argument will only be used for the option if you use it like: cmd --flag=value # or cmd -fvalue but not if used like: cmd --flag value # "value" here will be used as a positional argument # and "--flag" won't have an argument. If this weren't the case, using an option without an optional argument would be difficult if you also wanted to use positional arguments. For example: grep --color auto # Here "auto" will be used as the search string, # "color" will not have an argument and will fall back to the default, # which also *happens to be* auto. grep --color always # Here grep will still only use color "auto"matically # and search for the string "always". This isn't specific to argparse but common to all things using getopt(3) (if they have optional arguments at all). That grep example is how GNU grep actually behaves. Flag Value ValidationSometimes you need to validate the option values. For example, that it is a valid integer within a specific range, or an ip address, or something entirely different. You can always do this after argparse returns but you can also request that argparse perform the validation by executing arbitrary fish script. To do so simply append an ! (exclamation-mark) then the fish script to be run. When that code is executed three vars will be defined:
These variables are passed to the function as local exported variables. The script should write any error messages to stdout, not stderr. It should return a status of zero if the flag value is valid otherwise a non-zero status to indicate it is invalid. Fish ships with a _validate_int function that accepts a --min and --max flag. Let's say your command accepts a -m or --max flag and the minimum allowable value is zero and the maximum is 5. You would define the option like this: m/max=!_validate_int --min 0 --max 5. The default if you just call _validate_int without those flags is to simply check that the value is a valid integer with no limits on the min or max value allowed. Example OPTION_SPECsSome OPTION_SPEC examples:
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. begin - start a new block of codeSynopsisbegin; [COMMANDS...;] end Descriptionbegin is used to create a new block of code.A block allows the introduction of a new variable scope, redirection of the input or output of a set of commands as a group, or to specify precedence when using the conditional commands like and. The block is unconditionally executed. begin; ...; end is equivalent to if true; ...; end. begin does not change the current exit status itself. After the block has completed, $status will be set to the status returned by the most recent command. ExampleThe following code sets a number of variables inside of a block scope. Since the variables are set inside the block and have local scope, they will be automatically deleted when the block ends.begin set -l PIRATE Yarrr ... end echo $PIRATE # This will not output anything, since the PIRATE variable # went out of scope at the end of the block In the following code, all output is redirected to the file out.html. begin echo $xml_header echo $html_header if test -e $file ... end ... end > out.html bg - send jobs to backgroundSynopsisbg [PID...] Descriptionbg sends jobs to the background, resuming them if they are stopped.A background job is executed simultaneously with fish, and does not have access to the keyboard. If no job is specified, the last job to be used is put in the background. If PID is specified, the jobs containing the specified process IDs are put in the background. For compatibility with other shells, job expansion syntax is supported for bg. A PID of the format %1 will be interpreted as the PID of job 1. Job numbers can be seen in the output of jobs. When at least one of the arguments isn't a valid job specifier, bg will print an error without backgrounding anything. When all arguments are valid job specifiers, bg will background all matching jobs that exist. Examplebg 123 456 789 will background the jobs that contain processes 123, 456 and 789.If only 123 and 789 exist, it will still background them and print an error about 456. bg 123 banana or bg banana 123 will complain that "banana" is not a valid job specifier. bg %1 will background job 1. bind - handle fish key bindingsSynopsisbind [(-M | --mode) MODE] [(-m | --sets-mode) NEW_MODE] [--preset | --user] [(-s | --silent)] [(-k | --key)] SEQUENCE COMMAND [COMMAND...] bind [(-M | --mode) MODE] [(-k | --key)] [--preset] [--user] SEQUENCE bind (-K | --key-names) [(-a | --all)] [--preset] [--user] bind (-f | --function-names) bind (-L | --list-modes) bind (-e | --erase) [(-M | --mode) MODE] [--preset] [--user] (-a | --all | [(-k | --key)] SEQUENCE [SEQUENCE...]) Descriptionbind manages bindings.It can add bindings if given a SEQUENCE of characters to bind to. These should be written as fish escape sequences. The most important of these are \c for the control key, and \e for escape, and because of historical reasons also the Alt key (sometimes also called "Meta"). For example, Alt+W can be written as \ew, and Control+X (^X) can be written as \cx. Note that Alt-based key bindings are case sensitive and Control-based key bindings are not. This is a constraint of text-based terminals, not fish. The generic key binding that matches if no other binding does can be set by specifying a SEQUENCE of the empty string (that is, '' ). For most key bindings, it makes sense to bind this to the self-insert function (i.e. bind '' self-insert). This will insert any keystrokes not specifically bound to into the editor. Non-printable characters are ignored by the editor, so this will not result in control sequences being inserted. 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 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. When COMMAND is a shellscript command, it is a good practice to put the actual code into a 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. If a script produces output, it should finish by calling commandline -f repaint to tell fish that a repaint is in order. Note that special input functions cannot be combined with ordinary shell script commands. The commands must be entirely a sequence of special input functions (from bind -f) or all shell script commands (i.e., valid fish script). If no SEQUENCE is provided, all bindings (or just the bindings in the given MODE) are printed. If SEQUENCE is provided but no COMMAND, just the binding matching that sequence is printed. To save custom keybindings, put the bind statements into config.fish. Alternatively, fish also automatically executes a function called fish_user_key_bindings if it exists. Key bindings may use "modes", which mimics Vi's modal input behavior. The default mode is "default", and every bind applies to a single mode. The mode can be viewed/changed with the $fish_bind_mode variable. OptionsThe following options are available:
Special input functionsThe following special input functions are available:
Additional functionsThe following functions are included as normal functions, but are particularly useful for input editing:
ExamplesExit the shell when Control+D is pressed:bind \cd 'exit' Perform a history search when Page Up is pressed: bind -k ppage history-search-backward Turn on Vi key bindings and rebind Control+C to clear the input line: set -g fish_key_bindings fish_vi_key_bindings bind -M insert \cc kill-whole-line repaint Launch git diff and repaint the commandline afterwards when Control+G is pressed: bind \cg 'git diff; commandline -f repaint' Terminal LimitationsUnix terminals, like the ones fish operates in, are at heart 70s technology. They have some limitations that applications running inside them can't workaround.For instance, the control key modifies a character by setting the top three bits to 0. This means:
Other keys don't have a direct encoding, and are sent as escape sequences. For example → (Right) often sends \e\[C. These can differ from terminal to terminal, and the mapping is typically available in terminfo(5). Sometimes however a terminal identifies as e.g. xterm-256color for compatibility, but then implements xterm's sequences incorrectly. Special Case: The Escape CharacterThe escape key can be used standalone, for example, to switch from insertion mode to normal mode when using Vi keybindings. Escape can also be used as a "meta" key, to indicate the start of an escape sequence, like for function or arrow keys. Custom bindings can also be defined that begin with an escape character.Holding alt and something else also typically sends escape, for example holding alt+a will send an escape character and then an "a". fish waits for a period after receiving the escape character, to determine whether it is standalone or part of an escape sequence. While waiting, additional key presses make the escape key behave as a meta key. If no other key presses come in, it is handled as a standalone escape. The waiting period is set to 30 milliseconds (0.03 seconds). It can be configured by setting the fish_escape_delay_ms variable to a value between 10 and 5000 ms. This can be a universal variable that you set once from an interactive session. block - temporarily block delivery of eventsSynopsisblock [OPTIONS...] Descriptionblock prevents events triggered by fish or the emit command from being delivered and acted upon while the block is in place.In functions, block can be useful while performing work that should not be interrupted by the shell. The block can be removed. Any events which triggered while the block was in place will then be delivered. Event blocks should not be confused with code blocks, which are created with begin, if, while or for The following parameters are available:
Example# Create a function that listens for events function --on-event foo foo; echo 'foo fired'; end # Block the delivery of events block -g emit foo # No output will be produced block -e # 'foo fired' will now be printed NotesNote that events are only received from the current fish process as there is no way to send events from one fish process to another.break - stop the current inner loopSynopsisLOOP_CONSTRUCT; [COMMANDS...] break; [COMMANDS...] end Descriptionbreak halts a currently running loop, such as a switch, for or while loop. It is usually added inside of a conditional block such as an if block.There are no parameters for break. ExampleThe following code searches all .c files for "smurf", and halts at the first occurrence.for i in *.c if grep smurf $i echo Smurfs are present in $i break end end See Also
breakpoint - launch debug modeSynopsisbreakpoint Descriptionbreakpoint is used to halt a running script and launch an interactive debugging prompt.For more details, see Debugging fish scripts in the fish manual. There are no parameters for breakpoint. builtin - run a builtin commandSynopsisbuiltin [OPTIONS...] BUILTINNAME builtin --query BUILTINNAMES... Descriptionbuiltin forces the shell to use a builtin command, rather than a function or program.The following parameters are available:
Examplebuiltin jobs # executes the jobs builtin, even if a function named jobs exists case - conditionally execute a block of commandsSynopsisswitch VALUE; [case [WILDCARD...]; [COMMANDS...]; ...] end Descriptionswitch executes one of several blocks of commands, depending on whether a specified value matches one of several values. case is used together with the switch statement in order to determine which block should be executed.Each case command is given one or more parameters. The first case command with a parameter that matches the string specified in the switch command will be evaluated. case parameters may contain wildcards. These need to be escaped or quoted in order to avoid regular wildcard expansion using filenames. Note that fish does not fall through on case statements. Only the first matching case is executed. Note that command substitutions in a case statement will be evaluated even if its body is not taken. All substitutions, including command substitutions, must be performed before the value can be compared against the parameter. ExampleSay $animal contains the name of an animal. Then this code would classify it:switch $animal case cat echo evil case wolf dog human moose dolphin whale echo mammal case duck goose albatross echo bird case shark trout stingray echo fish # Note that the next case has a wildcard which is quoted case '*' echo I have no idea what a $animal is end If the above code was run with $animal set to whale, the output would be mammal. If $animal was set to "banana", it would print "I have no idea what a banana is". cd - change directorySynopsiscd [DIRECTORY] Descriptioncd changes the current working directory.If DIRECTORY is supplied, it will become the new directory. If no parameter is given, the contents of the HOME environment variable will be used. If DIRECTORY is a relative path, the paths found in the CDPATH list will be tried as prefixes for the specified path, in addition to $PWD. Note that the shell will attempt to change directory without requiring cd if the name of a directory is provided (starting with ., / or ~, or ending with /). Fish also ships a wrapper function around the builtin cd that understands cd - as changing to the previous directory. See also prevd. This wrapper function maintains a history of the 25 most recently visited directories in the $dirprev and $dirnext global variables. If you make those universal variables your cd history is shared among all fish instances. As a special case, cd . is equivalent to cd $PWD, which is useful in cases where a mountpoint has been recycled or a directory has been removed and recreated. Examplescd # changes the working directory to your home directory. cd /usr/src/fish-shell # changes the working directory to /usr/src/fish-shell See AlsoNavigate directories using the directory history or the directory stackcdh - change to a recently visited directorySynopsiscdh [ directory ] Descriptioncdh with no arguments presents a list of recently visited directories. You can then select one of the entries by letter or number. You can also press Tab to use the completion pager to select an item from the list. If you give it a single argument it is equivalent to cd directory.Note that the cd command limits directory history to the 25 most recently visited directories. The history is stored in the $dirprev and $dirnext variables which this command manipulates. If you make those universal variables your cd history is shared among all fish instances. See Also
command - run a programSynopsiscommand [OPTIONS] COMMANDNAME [ARGS...] Descriptioncommand forces the shell to execute the program COMMANDNAME and ignore any functions or builtins with the same name.The following options are available:
With the -s option, command treats every argument as a separate command to look up and sets the exit status to 0 if any of the specified commands were found, or 1 if no commands could be found. Additionally passing a -q or --quiet option prevents any paths from being printed, like type -q, for testing only the exit status. For basic compatibility with POSIX command, the -v flag is recognized as an alias for -s. Examplescommand ls causes fish to execute the ls program, even if an ls function exists.command -s ls returns the path to the ls program. command -q git; and command git log runs git log only if git exists. commandline - set or get the current command line bufferSynopsiscommandline [OPTIONS] [CMD] Descriptioncommandline can be used to set or get the current contents of the command line buffer.With no parameters, commandline returns the current value of the command line. With CMD specified, the command line buffer is erased and replaced with the contents of CMD. The following options are available:
The following options change the way commandline updates the command line buffer:
The following options change what part of the commandline is printed or updated:
The following options change the way commandline prints the current commandline buffer:
If commandline is called during a call to complete a given string using complete -C STRING, commandline will consider the specified string to be the current contents of the command line. The following options output metadata about the commandline state:
Examplecommandline -j $history[3] replaces the job under the cursor with the third item from the command line history.If the commandline contains >_ echo $flounder >&2 | less; and echo $catfish (with the cursor on the "o" of "flounder") The echo $flounder >& is the first process, less the second and and echo $catfish the third. echo $flounder >&2 | less is the first job, and echo $catfish the second. $flounder is the current token. More examples: >_ commandline -t $flounder >_ commandline -ct $fl >_ commandline -b # or just commandline echo $flounder >&2 | less; and echo $catfish >_ commandline -p echo $flounder >&2 >_ commandline -j echo $flounder >&2 | less complete - edit command specific tab-completionsSynopsiscomplete [( -c | --command | -p | --path )] COMMAND [( -c | --command | -p | --path ) COMMAND]... [( -e | --erase )] [( -s | --short-option ) SHORT_OPTION]... [( -l | --long-option | -o | --old-option ) LONG_OPTION]... [( -a | --arguments ) ARGUMENTS] [( -k | --keep-order )] [( -f | --no-files )] [( -F | --force-files )] [( -r | --require-parameter )] [( -x | --exclusive )] [( -w | --wraps ) WRAPPED_COMMAND]... [( -n | --condition ) CONDITION] [( -d | --description ) DESCRIPTION] complete ( -C [STRING] | --do-complete[=STRING] ) Descriptioncomplete defines, removes or lists completions for a command.For an introduction to writing your own completions, see Writing your own completions in the fish manual.
Command specific tab-completions in fish are based on the notion of options and arguments. An option is a parameter which begins with a hyphen, such as -h, -help or --help. Arguments are parameters that do not begin with a hyphen. Fish recognizes three styles of options, the same styles as the GNU getopt library. These styles are:
Multiple commands and paths can be given in one call to define the same completions for multiple commands. Multiple command switches and wrapped commands can also be given to define multiple completions in one call. Invoking complete multiple times for the same command adds the new definitions on top of any existing completions defined for the command. When -a or --arguments is specified in conjunction with long, short, or old style options, the specified arguments are only completed as arguments for any of the specified options. If -a or --arguments is specified without any long, short, or old style options, the specified arguments are used when completing non-option arguments to the command (except when completing an option argument that was specified with -r or --require-parameter). Command substitutions found in ARGUMENTS should return a newline-separated list of arguments, and each argument may optionally have a tab character followed by the argument description. Description given this way override a description given with -d or --description. The -w or --wraps options causes the specified command to inherit completions from another command, "wrapping" the other command. The wrapping command can also have additional completions. A command can wrap multiple commands, and wrapping is transitive: if A wraps B, and B wraps C, then A automatically inherits all of C's completions. Wrapping can be removed using the -e or --erase options. Wrapping only works for completions specified with -c or --command and are ignored when specifying completions with -p or --path. When erasing completions, it is possible to either erase all completions for a specific command by specifying complete -c COMMAND -e, or by specifying a specific completion option to delete. When complete is called without anything that would define or erase completions (options, arguments, wrapping, ...), it shows matching completions instead. So complete without any arguments shows all loaded completions, complete -c foo shows all loaded completions for foo. Since completions are autoloaded, you will have to trigger them first. ExamplesThe short style option -o for the gcc command needs a file argument:complete -c gcc -s o -r The short style option -d for the grep command requires one of read, skip or recurse: complete -c grep -s d -x -a "read skip recurse" The su command takes any username as an argument. Usernames are given as the first colon-separated field in the file /etc/passwd. This can be specified as: complete -x -c su -d "Username" -a "(cat /etc/passwd | cut -d : -f 1)" The rpm command has several different modes. If the -e or --erase flag has been specified, rpm should delete one or more packages, in which case several switches related to deleting packages are valid, like the nodeps switch. This can be written as: complete -c rpm -n "__fish_contains_opt -s e erase" -l nodeps -d "Don't check dependencies" where __fish_contains_opt is a function that checks the command line buffer for the presence of a specified set of options. To implement an alias, use the -w or --wraps option: complete -c hub -w git Now hub inherits all of the completions from git. Note this can also be specified in a function declaration (function thing -w otherthing). complete -c git Show all completions for git. contains - test if a word is present in a listSynopsiscontains [OPTIONS] KEY [VALUES...] Descriptioncontains tests whether the set VALUES contains the string KEY. If so, contains exits with status 0; if not, it exits with status 1.The following options are available:
Note that, like GNU tools and most of fish's builtins, contains interprets all arguments starting with a - as options to contains, until it reaches an argument that is -- (two dashes). See the examples below. ExampleIf $animals is a list of animals, the following will test if it contains a cat:if contains cat $animals echo Your animal list is evil! end This code will add some directories to $PATH if they aren't yet included: for i in ~/bin /usr/local/bin if not contains $i $PATH set PATH $PATH $i end end While this will check if hasargs was run with the -q option: function hasargs if contains -- -q $argv echo '$argv contains a -q option' end end The -- here stops contains from treating -q to an option to itself. Instead it treats it as a normal string to check. continue - skip the remainder of the current iteration of the current inner loopSynopsisLOOP_CONSTRUCT; [COMMANDS...;] continue; [COMMANDS...;] end Descriptioncontinue skips the remainder of the current iteration of the current inner loop, such as a for loop or a while loop. It is usually added inside of a conditional block such as an if statement or a switch statement.ExampleThe following code removes all tmp files that do not contain the word smurf.for i in *.tmp if grep smurf $i continue end # This "rm" is skipped over if "continue" is executed. rm $i # As is this "echo" echo $i end See Also
count - count the number of elements of a listSynopsiscount $VARIABLE COMMAND | count count < FILE Descriptioncount prints the number of arguments that were passed to it, plus the number of newlines passed to it via stdin. This is usually used to find out how many elements an environment variable list contains, or how many lines there are in a text file.count does not accept any options, not even -h or --help. count exits with a non-zero exit status if no arguments were passed to it, and with zero if at least one argument was passed. Note that, like wc -l, reading from stdin counts newlines, so echo -n foo | count will print 0. Examplecount $PATH # Returns the number of directories in the users PATH variable. count *.txt # Returns the number of files in the current working directory # ending with the suffix '.txt'. git ls-files --others --exclude-standard | count # Returns the number of untracked files in a git repository printf '%s\n' foo bar | count baz # Returns 3 (2 lines from stdin plus 1 argument) count < /etc/hosts # Counts the number of entries in the hosts file dirh - print directory historySynopsisdirh Descriptiondirh prints the current directory history. The current position in the history is highlighted using the color defined in the fish_color_history_current environment variable.dirh does not accept any parameters. Note that the cd command limits directory history to the 25 most recently visited directories. The history is stored in the $dirprev and $dirnext variables. See Also
dirs - print directory stackSynopsisdirs dirs -c Descriptiondirs prints the current directory stack, as created by pushd and modified by popd.With "-c", it clears the directory stack instead. dirs does not accept any parameters. See Also
disown - remove a process from the list of jobsSynopsisdisown [ PID ... ] Descriptiondisown removes the specified job from the list of jobs. The job itself continues to exist, but fish does not keep track of it any longer.Jobs in the list of jobs are sent a hang-up signal when fish terminates, which usually causes the job to terminate; disown allows these processes to continue regardless. If no process is specified, the most recently-used job is removed (like bg and fg). If one or more PIDs are specified, jobs with the specified process IDs are removed from the job list. Invalid jobs are ignored and a warning is printed. If a job is stopped, it is sent a signal to continue running, and a warning is printed. It is not possible to use the bg builtin to continue a job once it has been disowned. disown returns 0 if all specified jobs were disowned successfully, and 1 if any problems were encountered. Examplefirefox &; disown will start the Firefox web browser in the background and remove it from the job list, meaning it will not be closed when the fish process is closed.disown (jobs -p) removes all jobs from the job list without terminating them. echo - display a line of textSynopsisecho [OPTIONS] [STRING] Descriptionecho displays a string of text.The following options are available:
Unlike other shells, this echo accepts -- to signal the end of the options. Escape SequencesIf -e is used, the following sequences are recognized:
Example> echo 'Hello World' Hello World > echo -e 'Top\nBottom' Top Bottom > echo -- -n -n See Also
else - execute command if a condition is not metSynopsisif CONDITION; COMMANDS_TRUE...; [else; COMMANDS_FALSE...;] end Descriptionif will execute the command CONDITION. If the condition's exit status is 0, the commands COMMANDS_TRUE will execute. If it is not 0 and else is given, COMMANDS_FALSE will be executed.ExampleThe following code tests whether a file foo.txt exists as a regular file.if test -f foo.txt echo foo.txt exists else echo foo.txt does not exist end emit - emit a generic eventSynopsisemit EVENT_NAME [ARGUMENTS...] Descriptionemit emits, or fires, an event. Events are delivered to, or caught by, special functions called event handlers. The arguments are passed to the event handlers as function arguments.ExampleThe following code first defines an event handler for the generic event named 'test_event', and then emits an event of that type.function event_test --on-event test_event echo event test: $argv end emit test_event something NotesNote that events are only sent to the current fish process as there is no way to send events from one fish process to another.end - end a block of commandsSynopsisbegin; [COMMANDS...] end function NAME [OPTIONS]; COMMANDS...; end if CONDITION; COMMANDS_TRUE...; [else; COMMANDS_FALSE...;] end switch VALUE; [case [WILDCARD...]; [COMMANDS...]; ...] end while CONDITION; COMMANDS...; end for VARNAME in [VALUES...]; COMMANDS...; end Descriptionend ends a block of commands started by one of the following commands:
The end command does not change the current exit status. Instead, the status after it will be the status returned by the most recent command. eval - evaluate the specified commandsSynopsiseval [COMMANDS...] Descriptioneval evaluates the specified parameters as a command. If more than one parameter is specified, all parameters will be joined using a space character as a separator.If your command does not need access to stdin, consider using source instead. If no piping or other compound shell constructs are required, variable-expansion-as-command, as in set cmd ls -la; $cmd, is also an option. ExampleThe following code will call the ls command and truncate each filename to the first 12 characters.set cmd ls \| cut -c 1-12 eval $cmd exec - execute command in current processSynopsisexec COMMAND [OPTIONS...] Descriptionexec replaces the currently running shell with a new command. On successful completion, exec never returns. exec cannot be used inside a pipeline.Exampleexec emacs starts up the emacs text editor, and exits fish. When emacs exits, the session will terminate.exit - exit the shellSynopsisexit [STATUS] Descriptionexit causes fish to exit. If STATUS is supplied, it will be converted to an integer and used as the exit status. Otherwise, the exit status will be that of the last command executed.If exit is called while sourcing a file (using the source builtin) the rest of the file will be skipped, but the shell itself will not exit. false - return an unsuccessful resultSynopsisfalse Descriptionfalse sets the exit status to 1.See Also
fg - bring job to foregroundSynopsisfg [PID] Descriptionfg brings the specified job to the foreground, resuming it if it is stopped. While a foreground job is executed, fish is suspended. If no job is specified, the last job to be used is put in the foreground. If PID is specified, the job containing a process with the specified process ID is put in the foreground.For compatibility with other shells, job expansion syntax is supported for fg. A PID of the format %1 will foreground job 1. Job numbers can be seen in the output of jobs. Examplefg will put the last job in the foreground.fg %3 will put job 3 into the foreground. fish - the friendly interactive shellSynopsisfish [OPTIONS] [-c command] [FILE] [ARGUMENTS...] Descriptionfish is a command-line shell written mainly with interactive use in mind. This page briefly describes the options for invoking fish. The full manual is available in HTML by using the help command from inside fish, and in the fish-doc(1) man page. The tutorial is available as HTML via help tutorial or in fish-tutorial(1).The following options are available:
The fish exit status is generally the exit status of the last foreground command. DebuggingWhile fish provides extensive support for debugging fish scripts, it is also possible to debug and instrument its internals. Debugging can be enabled by passing the --debug option. For example, the following command turns on debugging for background IO thread events, in addition to the default categories, i.e. debug, error, warning, and warning-path:> fish --debug=iothread Available categories are listed by fish --print-debug-categories. The --debug option accepts a comma-separated list of categories, and supports glob syntax. The following command turns on debugging for complete, history, history-file, and profile-history, as well as the default categories: > fish --debug='complete,*history*' Debug messages output to stderr by default. Note that if fish_trace is set, execution tracing also outputs to stderr by default. You can output to a file using the --debug-output option: > fish --debug='complete,*history*' --debug-output=/tmp/fish.log --init-command='set fish_trace on' These options can also be changed via the $FISH_DEBUG and $FISH_DEBUG_OUTPUT variables. The categories enabled via --debug are added to the ones enabled by $FISH_DEBUG, so they can be disabled by prefixing them with - (reader-*,-ast* enables reader debugging and disables ast debugging). The file given in --debug-output takes precedence over the file in $FISH_DEBUG_OUTPUT. fish_add_path - add to the pathSynopsisfish_add_path [paths...] fish_add_path (-h | --help) fish_add_path [(-g | --global) | (-U | --universal) | (-P | --path)] [(-m | --move)] [(-a | --append) | (-p | --prepend)] [(-v | --verbose) | (-n | --dry-run)] [paths...] Descriptionfish_add_path is a simple way to add more components to fish's $PATH. It does this by adding the components either to $fish_user_paths or directly to $PATH (if the --path switch is given).It is (by default) safe to use fish_add_path in config.fish, or it can be used once, interactively, and the paths will stay in future because of universal variables. This is a "do what I mean" style command, if you need more control, consider modifying the variable yourself. Components are normalized by realpath. This means that trailing slashes are ignored and relative paths are made absolute (but symlinks are not resolved). If a component already exists, it is not added again and stays in the same place unless the --move switch is given. Components are added in the order they are given, and they are prepended to the path unless --append is given (if $fish_user_paths is used, that means they are last in $fish_user_paths, which is itself prepended to $PATH, so they still stay ahead of the system paths). If no component is new, the variable ($fish_user_paths or $PATH) is not set again or otherwise modified, so variable handlers are not triggered. If a component is not an existing directory, fish_add_path ignores it. Options
If --move is used, it may of course lead to the path swapping order, so you should be careful doing that in config.fish. Example# I just installed mycoolthing and need to add it to the path to use it. > fish_add_path /opt/mycoolthing/bin # I want my ~/.local/bin to be checked first. > fish_add_path -m ~/.local/bin # I prefer using a global fish_user_paths > fish_add_path -g ~/.local/bin ~/.otherbin /usr/local/sbin # I want to append to the entire $PATH because this directory contains fallbacks > fish_add_path -aP /opt/fallback/bin # I want to add the bin/ directory of my current $PWD (say /home/nemo/) > fish_add_path -v bin/ set fish_user_paths /home/nemo/bin /usr/bin /home/nemo/.local/bin # I have installed ruby via homebrew > fish_add_path /usr/local/opt/ruby/bin fish_breakpoint_prompt - define the prompt when stopped at a breakpointSynopsisfunction fish_breakpoint_prompt ... end Descriptionfish_breakpoint_prompt is the prompt function when asking for input in response to a breakpoint command.The exit status of commands within fish_breakpoint_prompt will not modify the value of $status outside of the fish_breakpoint_prompt function. fish ships with a default version of this function that displays the function name and line number of the current execution context. ExampleA simple prompt that is a simplified version of the default debugging prompt:function fish_breakpoint_prompt -d "Write out the debug prompt" set -l function (status current-function) set -l line (status current-line-number) set -l prompt "$function:$line >" echo -ns (set_color $fish_color_status) "BP $prompt" (set_color normal) ' ' end fish_command_not_found - what to do when a command wasn't foundSynopsisfunction fish_command_not_found ... end DescriptionWhen fish tries to execute a command and can't find it, it invokes this function.It can print a message to tell you about it, and it often also checks for a missing package that would include the command. Fish ships multiple handlers for various operating systems and chooses from them when this function is loaded, or you can define your own. It receives the full commandline as one argument per token, so $argv[1] contains the missing command. When you leave fish_command_not_found undefined (e.g. by adding an empty function file) or explicitly call __fish_default_command_not_found_handler, fish will just print a simple error. ExampleA simple handler:function fish_command_not_found echo Did not find command $argv[1] end > flounder Did not find command flounder Or the handler for OpenSUSE's command-not-found: function fish_command_not_found /usr/bin/command-not-found $argv[1] end Or the simple default handler: function fish_command_not_found __fish_default_command_not_found_handler $argv end Backwards compatibilityThis command was introduced in fish 3.2.0. Previous versions of fish used the "fish_command_not_found" event instead.To define a handler that works in older versions of fish as well, define it the old way: function __fish_command_not_found_handler --on-event fish_command_not_found echo COMMAND WAS NOT FOUND MY FRIEND $argv[1] end in which case fish will define a fish_command_not_found that calls it, or define a wrapper: function fish_command_not_found echo "G'day mate, could not find your command: $argv" end function __fish_command_not_found_handler --on-event fish_command_not_found fish_command_not_found $argv end fish_config - start the web-based configuration interfaceSynopsisfish_config fish_config browse fish_config prompt (choose | list | save | show) Descriptionfish_config is used to configure fish.Without arguments or with the browse command it starts the web-based configuration interface. The web interface allows you to view your functions, variables and history, and to make changes to your prompt and color configuration. It starts a local web server and opens a browser window. When you are finished, close the browser window and press the Enter key to terminate the configuration session. If the BROWSER environment variable is set, it will be used as the name of the web browser to open instead of the system default. With the prompt command fish_config can be used to view and choose a prompt from fish's sample prompts inside the terminal directly. Available subcommands for the prompt command:
Examplefish_config or fish_config browse opens a new web browser window and allows you to configure certain fish settings.fish_config prompt show demos the available sample prompts. fish_config prompt choose disco makes the disco prompt the prompt for the current session. This can also be used in config.fish to set the prompt. fish_config prompt save saves the current prompt to an autoloaded file. fish_config prompt save default chooses the default prompt and saves it. fish_git_prompt - output git information for use in a promptSynopsisfunction fish_prompt printf '%s' $PWD (fish_git_prompt) ' $ ' end DescriptionThe fish_git_prompt function displays information about the current git repository, if any.Git must be installed. There are numerous customization options, which can be controlled with git options or fish variables. git options, where available, take precedence over the fish variable with the same function. git options can be set on a per-repository or global basis. git options can be set with the git config command, while fish variables can be set as usual with the set command.
If none of these apply, the commit SHA shortened to 8 characters is used.
A number of variables set characters and color used as indicators. Many of these have a different default if used with informative status enabled, or $__fish_git_prompt_use_informative_chars set. The usual default is given first, then the informative default (if it is different). If no default for the colors is given, they default to $__fish_git_prompt_color.
Some variables are only used in some modes, like when informative status is enabled:
Variables used with showdirtystate:
Variables used with showstashstate:
Variables used with showuntrackedfiles:
Variables used with showupstream (also implied by informative status):
Colors used with showcolorhints:
Note that all colors can also have a corresponding _done color. For example, the contents of $__fish_git_prompt_color_upstream_done is printed right _after_ the upstream. See also fish_vcs_prompt, which will call all supported version control prompt functions, including git, Mercurial and Subversion. ExampleA simple prompt that displays git info:function fish_prompt # ... set -g __fish_git_prompt_showupstream auto printf '%s %s$' $PWD (fish_git_prompt) end fish_greeting - display a welcome message in interactive shellsSynopsisfunction fish_greeting ... end DescriptionWhen an interactive fish starts, it executes fish_greeting and displays its output.The default fish_greeting is a function that prints a variable of the same name ($fish_greeting), so you can also just change that if you just want to change the text. While you could also just put echo calls into config.fish, fish_greeting takes care of only being used in interactive shells, so it won't be used e.g. with scp (which executes a shell), which prevents some errors. ExampleA simple greeting:function fish_greeting echo Hello friend! echo The time is (set_color yellow; date +%T; set_color normal) and this machine is called $hostname end fish_hg_prompt - output Mercurial information for use in a promptSynopsisfunction fish_prompt printf '%s' $PWD (fish_hg_prompt) ' $ ' end DescriptionThe fish_hg_prompt function displays information about the current Mercurial repository, if any.Mercurial (hg) must be installed. 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 If you enabled the informative status, there are numerous customization options, which can be controlled with fish variables.
Some colors for status symbols:
The status symbols themselves:
Finally, $fish_prompt_hg_status_order, which can be used to change the order the status symbols appear in. It defaults to added modified copied deleted untracked unmerged. See also fish_vcs_prompt, which will call all supported version control prompt functions, including git, Mercurial and Subversion. ExampleA simple prompt that displays hg info:function fish_prompt ... set -g fish_prompt_hg_show_informative_status printf '%s %s$' $PWD (fish_hg_prompt) end fish_indent - indenter and prettifierSynopsisfish_indent [OPTIONS] [FILE...] Descriptionfish_indent is used to indent a piece of fish code. fish_indent reads commands from standard input or the given filenames and outputs them to standard output or a specified file (if -w is given).The following options are available:
fish_is_root_user - check if the current user is rootSynopsisfish_is_root_user Descriptionfish_is_root_user will check if the current user is root. It can be useful for the prompt to display something different if the user is root, for example.ExampleA simple example:function example --description 'Just an example' if fish_is_root_user do_something_different end end fish_key_reader - explore what characters keyboard keys sendSynopsisfish_key_reader [OPTIONS] Descriptionfish_key_reader is used to study input received from the terminal and can help with key binds. The program is interactive and works on standard input. Individual characters themselves and their hexadecimal values are displayed.The tool will write an example bind command matching the character sequence captured to stdout. If the character sequence matches a special key name (see bind --key-names), both bind CHARS ... and bind -k KEYNAME ... usage will be shown. Additional details about the characters received, such as the delay between chars, are written to stderr. The following options are available:
Usage NotesThe delay in milliseconds since the previous character was received is included in the diagnostic information written to stderr. This information may be useful to determine the optimal fish_escape_delay_ms setting or learn the amount of lag introduced by tools like ssh, mosh or tmux.fish_key_reader intentionally disables handling of many signals. To terminate fish_key_reader in --continuous mode do:
fish_mode_prompt - define the appearance of the mode indicatorSynopsisfunction fish_mode_prompt echo -n "$fish_bind_mode " end DescriptionThe fish_mode_prompt function outputs the mode indicator for use in vi-mode.The default fish_mode_prompt function will output indicators about the current Vi editor mode displayed to the left of the regular prompt. Define your own function to customize the appearance of the mode indicator. The $fish_bind_mode variable can be used to determine the current mode. It will be one of default, insert, replace_one, or visual. You can also define an empty fish_mode_prompt function to remove the Vi mode indicators: function fish_mode_prompt; end funcsave fish_mode_prompt fish_mode_prompt will be executed when the vi mode changes. If it produces any output, it is displayed and used. If it does not, the other prompt functions (fish_prompt and fish_right_prompt) will be executed as well in case they contain a mode display. Examplefunction fish_mode_prompt switch $fish_bind_mode case default set_color --bold red echo 'N' case insert set_color --bold green echo 'I' case replace_one set_color --bold green echo 'R' case visual set_color --bold brmagenta echo 'V' case '*' set_color --bold red echo '?' end set_color normal end Outputting multiple lines is not supported in fish_mode_prompt. fish_opt - create an option spec for the argparse commandSynopsisfish_opt [ -h | --help ] fish_opt ( -s X | --short=X ) [ -l LONG | --long=LONG ] [ --long-only ] [ -o | --optional-val ] [ -r | --required-val ] [ --multiple-vals ] DescriptionThis command provides a way to produce option specifications suitable for use with the argparse command. You can, of course, write the option specs by hand without using this command. But you might prefer to use this for the clarity it provides.The following argparse options are available:
ExamplesDefine a single option spec for the boolean help flag:set -l options (fish_opt -s h -l help) argparse $options -- $argv Same as above but with a second flag that requires a value: set -l options (fish_opt -s h -l help) set options $options (fish_opt -s m -l max --required-val) argparse $options -- $argv Same as above but with a third flag that can be given multiple times saving the value of each instance seen and only the long flag name (--token) can be used: set -l options (fish_opt --short=h --long=help) set options $options (fish_opt --short=m --long=max --required-val) set options $options (fish_opt --short=t --long=token --multiple-vals --long-only) argparse $options -- $argv fish_prompt - define the appearance of the command line promptSynopsisfunction fish_prompt ... end DescriptionThe fish_prompt function is executed when the prompt is to be shown, and the output is used as a prompt.The exit status of commands within fish_prompt will not modify the value of $status outside of the fish_prompt function. fish ships with a number of example prompts that can be chosen with the fish_config command. ExampleA simple prompt:function fish_prompt -d "Write out the prompt" # This shows up as USER@HOST /home/user/ >, with the directory colored # $USER and $hostname are set by fish, so you can just use them # instead of using `whoami` and `hostname` printf '%s@%s %s%s%s > ' $USER $hostname \ (set_color $fish_color_cwd) (prompt_pwd) (set_color normal) end fish_right_prompt - define the appearance of the right-side command line promptSynopsisfunction fish_right_prompt ... end Descriptionfish_right_prompt is similar to fish_prompt, except that it appears on the right side of the terminal window.Multiple lines are not supported in fish_right_prompt. ExampleA simple right prompt:function fish_right_prompt -d "Write out the right prompt" date '+%m/%d/%y' end fish_status_to_signal - Convert exit codes to human-friendly signalsSynopsisfunction fish_prompt echo -n (fish_status_to_signal $pipestatus | string join '|') (prompt_pwd) '$ ' end Descriptionfish_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 fish_svn_prompt - output Subversion information for use in a promptSynopsisfunction fish_prompt printf '%s' $PWD (fish_svn_prompt) ' $ ' end DescriptionThe fish_svn_prompt function displays information about the current Subversion repository, if any.Subversion (svn) must be installed. There are numerous customization options, which can be controlled with fish variables.
A number of variables control the symbol ("display") and color ("color") for the different status indicators:
See also fish_vcs_prompt, which will call all supported version control prompt functions, including git, Mercurial and Subversion. ExampleA simple prompt that displays svn info:function fish_prompt ... printf '%s %s$' $PWD (fish_svn_prompt) end fish_title - define the terminal's titleSynopsisfunction fish_title ... end DescriptionThe fish_title function is executed before and after a new command is executed or put into the foreground and the output is used as a titlebar message.The first argument to fish_title contains the most recently executed foreground command as a string, if any. This requires that your terminal supports programmable titles and the feature is turned on. ExampleA simple title:function fish_title set -q argv[1]; or set argv fish # Looks like ~/d/fish: git log # or /e/apt: fish echo (fish_prompt_pwd_dir_length=1 prompt_pwd): $argv; end fish_update_completions - update completions using manual pagesSynopsisfish_update_completions Descriptionfish_update_completions parses manual pages installed on the system, and attempts to create completion files in the fish configuration directory.This does not overwrite custom completions. There are no parameters for fish_update_completions. fish_vcs_prompt - output version control system information for use in a promptSynopsisfunction fish_prompt printf '%s' $PWD (fish_vcs_prompt) ' $ ' end DescriptionThe fish_vcs_prompt function displays information about the current version control system (VCS) repository, if any.It calls out to VCS-specific functions. The currently supported systems are:
If a VCS isn't installed, the respective function does nothing. The svn prompt is disabled by default because it's slow on large svn repositories. To enable it, modify fish_vcs_prompt to uncomment it. See funced. For more information, see the documentation for each of the functions above. ExampleA simple prompt that displays all known VCS info:function fish_prompt ... set -g __fish_git_prompt_showupstream auto printf '%s %s$' $PWD (fish_vcs_prompt) end for - perform a set of commands multiple timesSynopsisfor VARNAME in [VALUES...]; COMMANDS...; end Descriptionfor is a loop construct. It will perform the commands specified by COMMANDS multiple times. On each iteration, the local variable specified by VARNAME is assigned a new value from VALUES. If VALUES is empty, COMMANDS will not be executed at all. The VARNAME is visible when the loop terminates and will contain the last value assigned to it. If VARNAME does not already exist it will be set in the local scope. For our purposes if the for block is inside a function there must be a local variable with the same name. If the for block is not nested inside a function then global and universal variables of the same name will be used if they exist.Examplefor i in foo bar baz; echo $i; end # would output: foo bar baz NotesThe VARNAME was local to the for block in releases prior to 3.0.0. This means that if you did something like this:for var in a b c if break_from_loop break end end echo $var The last value assigned to var when the loop terminated would not be available outside the loop. What echo $var would write depended on what it was set to before the loop was run. Likely nothing. funced - edit a function interactivelySynopsisfunced [OPTIONS] NAME Descriptionfunced provides an interface to edit the definition of the function NAME.If the $VISUAL environment variable is set, it will be used as the program to edit the function. If $VISUAL is unset but $EDITOR is set, that will be used. Otherwise, a built-in editor will be used. Note that to enter a literal newline using the built-in editor you should press Alt+Enter. Pressing Enter signals that you are done editing the function. This does not apply to an external editor like emacs or vim. If there is no function called NAME a new function will be created with the specified name
ExampleSay you want to modify your prompt.Run: >_ funced fish_prompt This will open up your editor, allowing you to modify the function. When you're done, save and quit. Fish will reload the function, so you should see the changes right away. When you're done, use: >_ funcsave fish_prompt For more, see funcsave. funcsave - save the definition of a function to the user's autoload directorySynopsisfuncsave FUNCTION_NAME funcsave [(-d | --directory) where/to/save ] FUNCTION_NAME Descriptionfuncsave saves a function to a file in the fish configuration directory. This function will be automatically loaded by current and future fish sessions. This can be useful if you have interactively created a new function and wish to save it for later use.Note that because fish loads functions on-demand, saved functions will not function as event handlers until they are run or sourced otherwise. To activate an event handler for every new shell, add the function to your configuration file instead of using funcsave. This is typically used together with funced, which will open the function in your editor and load it in the current seession afterwards. function - create a functionSynopsisfunction NAME [OPTIONS]; BODY; end Descriptionfunction creates a new function NAME with the body BODY.A function is a list of commands that will be executed when the name of the function is given as a command. The following options are available:
If the user enters any additional arguments after the function, they are inserted into the environment variable list $argv. If the --argument-names option is provided, the arguments are also assigned to names specified in that option. By using one of the event handler switches, a function can be made to run automatically at specific events. The user may generate new events using the emit builtin. Fish generates the following named events:
Examplefunction ll ls -l $argv end will run the ls command, using the -l option, while passing on any additional files and switches to ls. function mkdir -d "Create a directory and set CWD" command mkdir $argv if test $status = 0 switch $argv[(count $argv)] case '-*' case '*' cd $argv[(count $argv)] return end end end This will run the mkdir command, and if it is successful, change the current working directory to the one just created. function notify set -l job (jobs -l -g) or begin; echo "There are no jobs" >&2; return 1; end function _notify_job_$job --on-job-exit $job --inherit-variable job echo -n \a # beep functions -e _notify_job_$job end end This will beep when the most recent job completes. NotesEvents are only received from the current fish process as there is no way to send events from one fish process to another.See moreFor more explanation of how functions fit into fish, see Functions.functions - print or erase functionsSynopsisfunctions [ -a | --all ] [ -n | --names ] functions [ -D | --details ] [ -v ] FUNCTION functions -c OLDNAME NEWNAME functions -d DESCRIPTION FUNCTION functions [ -e | -q ] FUNCTIONS... Descriptionfunctions prints or erases functions.The following options are available:
You should not assume that only five lines will be written since we may add additional information to the output in the future.
The default behavior of functions, when called with no arguments, is to print the names of all defined functions. Unless the -a option is given, no functions starting with underscores are included in the output. If any non-option parameters are given, the definition of the specified functions are printed. Copying a function using -c copies only the body of the function, and does not attach any event notifications from the original function. Only one function's description can be changed in a single invocation of functions -d. The exit status of functions is the number of functions specified in the argument list that do not exist, which can be used in concert with the -q option. Examplesfunctions -n # Displays a list of currently-defined functions functions -c foo bar # Copies the 'foo' function to a new function called 'bar' functions -e bar # Erases the function ``bar`` See moreFor more explanation of how functions fit into fish, see Functions.help - display fish documentationSynopsishelp [SECTION] Descriptionhelp displays the fish help documentation.If a SECTION is specified, the help for that command is shown. If the BROWSER environment variable is set, it will be used to display the documentation. Otherwise, fish will search for a suitable browser. If you prefer to use a different browser (other than as described above) for fish help, you can set the fish_help_browser variable. This variable may be set as a list, where the first element is the browser command and the rest are browser options. Note that most builtin commands display their help in the terminal when given the --help option. Examplehelp fg shows the documentation for the fg builtin.history - show and manipulate command historySynopsishistory [ search ] [ --show-time ] [ --case-sensitive ] [ --exact | --prefix | --contains ] [ --max=n ] [ --null ] [ -R | --reverse ] [ "search string"... ] history delete [ --show-time ] [ --case-sensitive ] [ --exact | --prefix | --contains ] "search string"... history merge history save history clear history ( -h | --help ) Descriptionhistory is used to search, delete, and otherwise manipulate the history of interactive commands.The following operations (sub-commands) are available:
The following options are available: These flags can appear before or immediately after one of the sub-commands listed above.
Examplehistory clear # Deletes all history items history search --contains "foo" # Outputs a list of all previous commands containing the string "foo". history delete --prefix "foo" # Interactively deletes commands which start with "foo" from the history. # You can select more than one entry by entering their IDs separated by a space. Customizing the name of the history fileBy default interactive commands are logged to $XDG_DATA_HOME/fish/fish_history (typically ~/.local/share/fish/fish_history).You can set the fish_history variable to another name for the current shell session. The default value (when the variable is unset) is fish which corresponds to $XDG_DATA_HOME/fish/fish_history. If you set it to e.g. fun, the history would be written to $XDG_DATA_HOME/fish/fun_history. An empty string means history will not be stored at all. This is similar to the private session features in web browsers. You can change fish_history at any time (by using set -x fish_history "session_name") and it will take effect right away. If you set it to "default", it will use the default session name (which is "fish"). Other shells such as bash and zsh use a variable named HISTFILE for a similar purpose. Fish uses a different name to avoid conflicts and signal that the behavior is different (session name instead of a file path). Also, if you set the var to anything other than fish or default it will inhibit importing the bash history. That's because the most common use case for this feature is to avoid leaking private or sensitive history when giving a presentation. NotesIf you specify both --prefix and --contains the last flag seen is used.Note that for backwards compatibility each subcommand can also be specified as a long option. For example, rather than history search you can type history --search. Those long options are deprecated and will be removed in a future release. if - conditionally execute a commandSynopsisif CONDITION; COMMANDS_TRUE...; [else if CONDITION2; COMMANDS_TRUE2...;] [else; COMMANDS_FALSE...;] end Descriptionif will execute the command CONDITION. If the condition's exit status is 0, the commands COMMANDS_TRUE will execute. If the exit status is not 0 and else is given, COMMANDS_FALSE will be executed.You can use and or or in the condition. See the second example below. The exit status of the last foreground command to exit can always be accessed using the $status variable. ExampleThe following code will print foo.txt exists if the file foo.txt exists and is a regular file, otherwise it will print bar.txt exists if the file bar.txt exists and is a regular file, otherwise it will print foo.txt and bar.txt do not exist.if test -f foo.txt echo foo.txt exists else if test -f bar.txt echo bar.txt exists else echo foo.txt and bar.txt do not exist end The following code will print "foo.txt exists and is readable" if foo.txt is a regular file and readable if test -f foo.txt and test -r foo.txt echo "foo.txt exists and is readable" end isatty - test if a file descriptor is a terminalSynopsisisatty [FILE DESCRIPTOR] Descriptionisatty tests if a file descriptor is a terminal (as opposed to a file). The name is derived from the system call of the same name, which for historical reasons refers to a teletypewriter (TTY).FILE DESCRIPTOR may be either the number of a file descriptor, or one of the strings stdin, stdout, or stderr. If not specified, zero is assumed. If the specified file descriptor is a terminal device, the exit status of the command is zero. Otherwise, the exit status is non-zero. No messages are printed to standard error. ExamplesFrom an interactive shell, the commands below exit with a return value of zero:isatty isatty stdout isatty 2 echo | isatty 1 And these will exit non-zero: echo | isatty isatty 9 isatty stdout > file isatty 2 2> file jobs - print currently running jobsSynopsisjobs [OPTIONS] [ PID | %JOBID ] Descriptionjobs prints a list of the currently running jobs and their status.jobs accepts the following switches:
On systems that supports this feature, jobs will print the CPU usage of each job since the last command was executed. The CPU usage is expressed as a percentage of full CPU activity. Note that on multiprocessor systems, the total activity may be more than 100%. Arguments of the form PID or %JOBID restrict the output to jobs with the selected process identifiers or job numbers respectively. If the output of jobs is redirected or if it is part of a command substitution, the column header that is usually printed is omitted, making it easier to parse. The exit status of jobs is 0 if there are running background jobs and 1 otherwise. Examplejobs outputs a summary of the current jobs, such as two long-running tasks in this example:Job Group State Command 2 26012 running nc -l 55232 < /dev/random & 1 26011 running python tests/test_11.py & math - perform mathematics calculationsSynopsismath [-sN | --scale=N] [-bBASE | --base=BASE] [--] EXPRESSION Descriptionmath performs mathematical calculations. It supports simple operations such as addition, subtraction, and so on, as well as functions like abs(), sqrt() and ln().By default, the output is a floating-point number with trailing zeroes trimmed. To get a fixed representation, the --scale option can be used, including --scale=0 for integer output. Keep in mind that parameter expansion happens before expressions are evaluated. This can be very useful in order to perform calculations involving shell variables or the output of command substitutions, but it also means that parenthesis (()) and the asterisk (*) glob character have to be escaped or quoted. x can also be used to denote multiplication, but it needs to be followed by whitespace to distinguish it from hexadecimal numbers. Parentheses for functions are optional - math sin pi prints 0. However, a comma will bind to the inner function, so math pow sin 3, 5 is an error because it tries to give sin the arguments 3 and 5. When in doubt, use parentheses. math ignores whitespace between arguments and takes its input as multiple arguments (internally joined with a space), so math 2 +2 and math "2 + 2" work the same. math 2 2 is an error. The following options are available:
Return ValuesIf the expression is successfully evaluated and doesn't over/underflow or return NaN the return status is zero (success) else one.Syntaxmath knows some operators, constants, functions and can (obviously) read numbers.For numbers, . is always the radix character regardless of locale - 2.5, not 2,5. Scientific notation (10e5) and hexadecimal (0xFF) are also available. Operatorsmath knows the following operators:
They are all used in an infix manner - 5 + 2, not + 5 2. Constantsmath knows the following constants:
Use them without a leading $ - pi - 3 should be about 0. Functionsmath supports the following functions:
All of the trigonometric functions use radians (the pi-based scale, not 360°). Examplesmath 1+1 outputs 2.math $status - 128 outputs the numerical exit status of the last command minus 128. math 10 / 6 outputs 1.666667. math -s0 10.0 / 6.0 outputs 1. math -s3 10 / 6 outputs 1.666. math "sin(pi)" outputs 0. math 5 \* 2 or math "5 * 2" or math 5 "*" 2 all output 10. math 0xFF outputs 255, math 0 x 3 outputs 0 (because it computes 0 multiplied by 3). math bitand 0xFE, 0x2e outputs 46. math "bitor(9,2)" outputs 11. math --base=hex 192 prints 0xc0. math 'ncr(49,6)' prints 13983816 - that's the number of possible picks in 6-from-49 lotto. Compatibility notesFish 1.x and 2.x releases relied on the bc command for handling math expressions. Starting with fish 3.0.0 fish uses the tinyexpr library and evaluates the expression without the involvement of any external commands.You don't need to use -- before the expression, even if it begins with a minus sign which might otherwise be interpreted as an invalid option. If you do insert -- before the expression, it will cause option scanning to stop just like for every other command and it won't be part of the expression. nextd - move forward through directory historySynopsisnextd [ -l | --list ] [POS] Descriptionnextd moves forwards POS positions in the history of visited directories; if the end of the history has been hit, a warning is printed.If the -l or --list flag is specified, the current directory history is also displayed. Note that the cd command limits directory history to the 25 most recently visited directories. The history is stored in the $dirprev and $dirnext variables which this command manipulates. Examplecd /usr/src # Working directory is now /usr/src cd /usr/src/fish-shell # Working directory is now /usr/src/fish-shell prevd # Working directory is now /usr/src nextd # Working directory is now /usr/src/fish-shell See Also
not - negate the exit status of a jobSynopsisnot COMMAND [OPTIONS...] Descriptionnot negates the exit status of another command. If the exit status is zero, not returns 1. Otherwise, not returns 0.ExampleThe following code reports an error and exits if no file named spoon can be found.if not test -f spoon echo There is no spoon exit 1 end open - open file in its default applicationSynopsisopen FILES... Descriptionopen opens a file in its default application, using the appropriate tool for the operating system. On GNU/Linux, this requires the common but optional xdg-open utility, from the xdg-utils package.Note that this function will not be used if a command by this name exists (which is the case on macOS or Haiku). Exampleopen *.txt opens all the text files in the current directory using your system's default text editor.or - conditionally execute a commandSynopsisCOMMAND1; or COMMAND2 Descriptionor is used to execute a command if the previous command was not successful (returned a status of something other than 0).or statements may be used as part of the condition in an and or while block. or does not change the current exit status itself, but the command it runs most likely will. The exit status of the last foreground command to exit can always be accessed using the $status variable. ExampleThe following code runs the make command to build a program. If the build succeeds, the program is installed. If either step fails, make clean is run, which removes the files created by the build process.make; and make install; or make clean See Also
popd - move through directory stackSynopsispopd Descriptionpopd removes the top directory from the directory stack and changes the working directory to the new top directory. Use pushd to add directories to the stack.Examplepushd /usr/src # Working directory is now /usr/src # Directory stack contains /usr/src pushd /usr/src/fish-shell # Working directory is now /usr/src/fish-shell # Directory stack contains /usr/src /usr/src/fish-shell popd # Working directory is now /usr/src # Directory stack contains /usr/src See Also
prevd - move backward through directory historySynopsisprevd [ -l | --list ] [POS] Descriptionprevd moves backwards POS positions in the history of visited directories; if the beginning of the history has been hit, a warning is printed.If the -l or --list flag is specified, the current history is also displayed. Note that the cd command limits directory history to the 25 most recently visited directories. The history is stored in the $dirprev and $dirnext variables which this command manipulates. Examplecd /usr/src # Working directory is now /usr/src cd /usr/src/fish-shell # Working directory is now /usr/src/fish-shell prevd # Working directory is now /usr/src nextd # Working directory is now /usr/src/fish-shell See Also
printf - display text according to a format stringSynopsisprintf FORMAT [ARGUMENT ...] Descriptionprintf uses the format string FORMAT to print the ARGUMENT arguments. This means that it takes format specifiers in the format string and replaces each with an argument.The format argument is re-used as many times as necessary to convert all of the given arguments. So printf %s\n flounder catfish clownfish shark will print four lines. Unlike echo, printf does not append a new line unless it is specified as part of the string. It doesn't support any options, so there is no need for a -- separator, which makes it easier to use for arbitrary input than echo. [1] Format SpecifiersValid format specifiers are taken from the C library function printf(3):
%% signifies a literal "%". Conversion can fail, e.g. "102.234" can't losslessly convert to an integer, causing printf to print an error. If you are okay with losing information, silence errors with 2>/dev/null. A number between the % and the format letter specifies the width. The result will be left-padded with spaces. Backslash Escapesprintf also knows a number of backslash escapes:
Errors and Return StatusIf the given argument doesn't work for the given format (like when you try to convert a number like 3.141592 to an integer), printf prints an error, to stderr. printf will then also return non-zero, but will still try to print as much as it can.It will also return non-zero if no argument at all was given, in which case it will print nothing. This printf has been imported from the printf in GNU Coreutils version 6.9. If you would like to use a newer version of printf, for example the one shipped with your OS, try command printf. Exampleprintf '%s\t%s\n' flounder fish Will print "flounder fish" (separated with a tab character), followed by a newline character. This is useful for writing completions, as fish expects completion scripts to output the option followed by the description, separated with a tab character. printf '%s: %d' "Number of bananas in my pocket" 42 Will print "Number of bananas in my pocket: 42", without a newline. See Also
Footnotes
prompt_login - describe the login suitable for promptSynopsisfunction fish_prompt echo -n (prompt_login) (prompt_pwd) '$ ' end Descriptionprompt_login is a function to describe the current login. It will show the user, the host and also whether the shell is running in a chroot (currently debian's debian_chroot is supported).Examples>_ prompt_login root@bananablaster prompt_pwd - print pwd suitable for promptSynopsisfunction fish_prompt echo -n (prompt_pwd) '$ ' end Descriptionprompt_pwd is a function to print the current working directory in a way suitable for prompts. It will replace the home directory with "~" and shorten every path component but the last to a default of one character.To change the number of characters per path component, set $fish_prompt_pwd_dir_length to the number of characters. Setting it to 0 or an invalid value will disable shortening entirely. Examples>_ cd ~/ >_ echo $PWD /home/alfa >_ prompt_pwd ~ >_ cd /tmp/banana/sausage/with/mustard >_ prompt_pwd /t/b/s/w/mustard >_ set -g fish_prompt_pwd_dir_length 3 >_ prompt_pwd /tmp/ban/sau/wit/mustard psub - perform process substitutionSynopsisCOMMAND1 ( COMMAND2 | psub [-F | --fifo] [-f | --file] [-s SUFFIX]) DescriptionSome shells (e.g., ksh, bash) feature a syntax that is a mix between command substitution and piping, called process substitution. It is used to send the output of a command into the calling command, much like command substitution, but with the difference that the output is not sent through commandline arguments but through a named pipe, with the filename of the named pipe sent as an argument to the calling program. psub combined with a regular command substitution provides the same functionality.The following options are available:
Examplediff (sort a.txt | psub) (sort b.txt | psub) # shows the difference between the sorted versions of files ``a.txt`` and ``b.txt``. source-highlight -f esc (cpp main.c | psub -f -s .c) # highlights ``main.c`` after preprocessing as a C source. pushd - push directory to directory stackSynopsispushd [DIRECTORY] DescriptionThe pushd function adds DIRECTORY to the top of the directory stack and makes it the current working directory. popd will pop it off and return to the original directory.Without arguments, it exchanges the top two directories in the stack. pushd +NUMBER rotates the stack counter-clockwise i.e. from bottom to top pushd -NUMBER rotates clockwise i.e. top to bottom. Examplecd ~/dir1 pushd ~/dir2 pushd ~/dir3 # Working directory is now ~/dir3 # Directory stack contains ~/dir2 ~/dir1 pushd /tmp # Working directory is now /tmp # Directory stack contains ~/dir3 ~/dir2 ~/dir1 pushd +1 # Working directory is now ~/dir3 # Directory stack contains ~/dir2 ~/dir1 /tmp popd # Working directory is now ~/dir2 # Directory stack contains ~/dir1 /tmp See Also
pwd - output the current working directorySynopsispwd [(-P | --physical)] [(-L | --logical)] Descriptionpwd outputs (prints) the current working directory.The following options are available:
See AlsoNavigate directories using the directory history or the directory stackrandom - generate random numberSynopsisrandom random SEED random START END random START STEP END random choice [ITEMS...] Descriptionrandom generates a pseudo-random integer from a uniform distribution. The range (inclusive) depends on the arguments. No arguments indicate a range of 0 to 32767 (inclusive).If one argument is specified, the internal engine will be seeded with the argument for future invocations of random and no output will be produced. Two arguments indicate a range from START to END (both START and END included). Three arguments indicate a range from START to END with a spacing of STEP between possible outputs. random choice will select one random item from the succeeding arguments. Note that seeding the engine will NOT give the same result across different systems. You should not consider random cryptographically secure, or even statistically accurate. ExampleThe following code will count down from a random even number between 10 and 20 to 1:for i in (seq (random 10 2 20) -1 1) echo $i end And this will open a random picture from any of the subdirectories: open (random choice **.jpg) Or, to only get even numbers from 2 to 20: random 2 2 20 Or odd numbers from 1 to 3: random 1 2 3 # or 1 2 4 read - read line of input into variablesSynopsisread [OPTIONS] [VARIABLE ...] Descriptionread reads from standard input and either writes the result back to standard output (for use in command substitution), or stores the result in one or more shell variables. By default, read reads a single line and splits it into variables on spaces or tabs. Alternatively, a null character or a maximum number of characters can be used to terminate the input, and other delimiters can be given. Unlike other shells, there is no default variable (such as REPLY) for storing the result - instead, it is printed on standard output.The following options are available:
Without the --line option, read reads a single line of input from standard input, breaks it into tokens, and then assigns one token to each variable specified in VARIABLES. If there are more tokens than variables, the complete remainder is assigned to the last variable. If no option to determine how to split like --delimiter, --line or --tokenize is given, the variable IFS is used as a list of characters to split on. Relying on the use of IFS is deprecated and this behaviour will be removed in future versions. The default value of IFS contains space, tab and newline characters. As a special case, if IFS is set to the empty string, each character of the input is considered a separate token. With the --line option, read reads a line of input from standard input into each provided variable, stopping when each variable has been filled. The line is not tokenized. If no variable names are provided, read enters a special case that simply provides redirection from standard input to standard output, useful for command substitution. For instance, the fish shell command below can be used to read data that should be provided via a command line argument from the console instead of hardcoding it in the command itself, allowing the command to both be reused as-is in various contexts with different input values and preventing possibly sensitive text from being included in the shell history: mysql -uuser -p(read) When running in this mode, read does not split the input in any way and text is redirected to standard output without any further processing or manipulation. If -a or --array is provided, only one variable name is allowed and the tokens are stored as a list in this variable. See the documentation for set for more details on the scoping rules for variables. When read reaches the end-of-file (EOF) instead of the terminator, the exit status is set to 1. Otherwise, it is set to 0. In order to protect the shell from consuming too many system resources, read will only consume a maximum of 100 MiB (104857600 bytes); if the terminator is not reached before this limit then VARIABLE is set to empty and the exit status is set to 122. This limit can be altered with the fish_read_limit variable. If set to 0 (zero), the limit is removed. ExampleThe following code stores the value 'hello' in the shell variable $foo.echo hello|read foo # This is a neat way to handle command output by-line: printf '%s\n' line1 line2 line3 line4 | while read -l foo echo "This is another line: $foo" end # Delimiters given via "-d" are taken as one string echo a==b==c | read -d == -l a b c echo $a # a echo $b # b echo $c # c # --tokenize honors quotes and escaping like the shell's argument passing: echo 'a\ b' | read -t first second echo $first # outputs "a b", $second is empty echo 'a"foo bar"b (command echo wurst)*" "{a,b}' | read -lt -l a b c echo $a # outputs 'afoo bar' (without the quotes) echo $b # outputs '(command echo wurst)* {a,b}' (without the quotes) echo $c # nothing realpath - convert a path to an absolute path without symlinksSynopsisrealpath PATH Descriptionrealpath resolves a path to its absolute path.fish provides a realpath builtin as a fallback for systems where there is no realpath command, your OS might provide a version with more features. If a realpath command exists, it will be preferred, so if you want to use the builtin you should use builtin realpath explicitly. The following options are available:
return - stop the current inner functionSynopsisfunction NAME; [COMMANDS...;] return [STATUS]; [COMMANDS...;] end Descriptionreturn halts a currently running function. The exit status is set to STATUS if it is given.It is usually added inside of a conditional block such as an if statement or a switch statement to conditionally stop the executing function and return to the caller, but it can also be used to specify the exit status of a function. ExampleThe following code is an implementation of the false command as a fish functionfunction false return 1 end set - display and change shell variablesSynopsisset [SCOPE_OPTIONS] set [OPTIONS] VARIABLE_NAME VALUES... set [OPTIONS] VARIABLE_NAME[INDICES]... VALUES... set ( -q | --query ) [SCOPE_OPTIONS] VARIABLE_NAMES... set ( -e | --erase ) [SCOPE_OPTIONS] VARIABLE_NAME... set ( -e | --erase ) [SCOPE_OPTIONS] VARIABLE_NAME[INDICES]... set ( -S | --show ) [VARIABLE_NAME]... Descriptionset manipulates shell variables.If both a variable name and values are provided, set assigns the values to the variable of that name. Because all variables in fish are lists, multiple values are allowed. If only a variable name has been given, set sets the variable to the empty list. If set is called with no arguments, it prints the names and values of all shell variables in sorted order. Passing scope or export flags allows filtering this to only matching variables, so set --local would only show local variables. With --erase and optionally a scope flag set will erase the matching variable (or the variable of that name in the smallest possible scope). With --show, set will describe the given variable names, explaining how they have been defined - in which scope with which values and options. The following options control variable scope:
These options control additional variable options:
The following other options are available:
If a variable is set to more than one value, the variable will be a list with the specified elements. If a variable is set to zero elements, it will become a list with zero elements. If the variable name is one or more list elements, such as PATH[1 3 7], only those list elements specified will be changed. If you specify a negative index when expanding or assigning to a list variable, the index will be calculated from the end of the list. For example, the index -1 means the last index of a list. The scoping rules when creating or updating a variable are:
The exporting rules when creating or updating a variable are identical to the scoping rules for variables:
In query mode, the scope to be examined can be specified. In erase mode, if variable indices are specified, only the specified slices of the list variable will be erased. set requires all options to come before any other arguments. For example, set flags -l will have the effect of setting the value of the variable flags to '-l', not making the variable local. Exit statusIn assignment mode, set does not modify the exit status, but passes along whatever $status was set, including by command substitutions. This allows capturing the output and exit status of a subcommand, like in if set output (command).In query mode, the exit status is the number of variables that were not found. In erase mode, set exits with a zero exit status in case of success, with a non-zero exit status if the commandline was invalid, if any of the variables did not exist or was a special read-only variable. Examples# Prints all global, exported variables. set -xg # Sets the value of the variable $foo to be 'hi'. set foo hi # Appends the value "there" to the variable $foo. set -a foo there # Does the same thing as the previous two commands the way it would be done pre-fish 3.0. set foo hi set foo $foo there # Removes the variable $smurf set -e smurf # Changes the fourth element of the $PATH list to ~/bin set PATH[4] ~/bin # Outputs the path to Python if ``type -p`` returns true. if set python_path (type -p python) echo "Python is at $python_path" end # Setting a variable doesn't modify $status! false set foo bar echo $status # prints 1, because of the "false" above. true set foo banana (false) echo $status # prints 1, because of the "(false)" above. # Like other shells, pass a variable to just one command: # Run fish with a temporary home directory. HOME=(mktemp -d) fish # Which is essentially the same as: begin; set -lx HOME (mktemp -d); fish; end NotesFish versions prior to 3.0 supported the syntax set PATH[1] PATH[4] /bin /sbin, which worked like set PATH[1 4] /bin /sbin. This syntax was not widely used, and was ambiguous and inconsistent.set_color - set the terminal colorSynopsisset_color [OPTIONS] VALUE Descriptionset_color is used to control the color and styling of text in the terminal. VALUE describes that styling. It's a reserved color name like red or a RGB color value given as 3 or 6 hexadecimal digits ("F27" or "FF2277"). A special keyword normal resets text formatting to terminal defaults.Valid colors include:
The br- (as in 'bright') forms are full-brightness variants of the 8 standard-brightness colors on many terminals. brblack has higher brightness than black - towards gray. An RGB value with three or six hex digits, such as A0FF33 or f2f can be used. fish will choose the closest supported color. A three digit value is equivalent to specifying each digit twice; e.g., set_color 2BC is the same as set_color 22BBCC. Hexadecimal RGB values can be in lower or uppercase. Depending on the capabilities of your terminal (and the level of support set_color has for it) the actual color may be approximated by a nearby matching reserved color name or set_color may not have an effect on color. A second color may be given as a desired fallback color. e.g. set_color 124212 brblue will instruct set_color to use brblue if a terminal is not capable of the exact shade of grey desired. This is very useful when an 8 or 16 color terminal might otherwise not use a color. The following options are available:
Using the normal keyword will reset foreground, background, and all formatting back to default. Notes
Examplesset_color red; echo "Roses are red" set_color blue; echo "Violets are blue" set_color 62A; echo "Eggplants are dark purple" set_color normal; echo "Normal is nice" # Resets the background too Terminal Capability DetectionFish uses some heuristics to determine what colors a terminal supports to avoid sending sequences that it won't understand.In particular it will:
If terminfo reports 256 color support for a terminal, 256 color support will always be enabled. To force true-color support on or off, set $fish_term24bit to "1" for on and 0 for off - set -g fish_term24bit 1. To debug color palette problems, tput colors may be useful to see the number of colors in terminfo for a terminal. Fish launched as fish -d2 will include diagnostic messages that indicate the color support mode in use. The set_color command uses the terminfo database to look up how to change terminal colors on whatever terminal is in use. Some systems have old and incomplete terminfo databases, and lack color information for terminals that support it. Fish assumes that all terminals can use the [ANSI X3.64](https://en.wikipedia.org/wiki/ANSI_escape_code) escape sequences if the terminfo definition indicates a color below 16 is not supported. source - evaluate contents of fileSynopsissource FILENAME [ARGUMENTS...] somecommand | source Descriptionsource evaluates the commands of the specified file in the current shell as a new block of code. This is different from starting a new process to perform the commands (i.e. fish < FILENAME) since the commands will be evaluated by the current shell, which means that changes in shell variables will affect the current shell. If additional arguments are specified after the file name, they will be inserted into the $argv variable. The $argv variable will not include the name of the sourced file.fish will search the working directory to resolve relative paths but will not search $PATH. If no file is specified and stdin is not the terminal, or if the file name - is used, stdin will be read. The exit status of source is the exit status of the last job to execute. If something goes wrong while opening or reading the file, source exits with a non-zero status. . (a single period) is an alias for the source command. The use of . is deprecated in favour of source, and . will be removed in a future version of fish. source creates a new local scope; set --local within a sourced block will not affect variables in the enclosing scope. Examplesource ~/.config/fish/config.fish # Causes fish to re-read its initialization file. CaveatsIn fish versions prior to 2.3.0, the $argv variable would have a single element (the name of the sourced file) if no arguments are present. Otherwise, it would contain arguments without the name of the sourced file. That behavior was very confusing and unlike other shells such as bash and zsh.status - query fish runtime informationSynopsisstatus status is-login status is-interactive status is-block status is-breakpoint status is-command-substitution status is-no-job-control status is-full-job-control status is-interactive-job-control status current-command status filename status basename status dirname status fish-path status function status line-number status stack-trace status job-control CONTROL_TYPE status features status test-feature FEATURE DescriptionWith no arguments, status displays a summary of the current login and job control status of the shell.The following operations (sub-commands) are available:
NotesFor backwards compatibility most subcommands can also be specified as a long or short option. For example, rather than status is-login you can type status --is-login. The flag forms are deprecated and may be removed in a future release (but not before fish 4.0).You can only specify one subcommand per invocation even if you use the flag form of the subcommand. string - manipulate stringsSynopsisstring collect [(-N | --no-trim-newlines)] [STRING...] string escape [(-n | --no-quoted)] [--style=xxx] [STRING...] string join [(-q | --quiet)] SEP [STRING...] string join0 [(-q | --quiet)] [STRING...] string length [(-q | --quiet)] [STRING...] string lower [(-q | --quiet)] [STRING...] string match [(-a | --all)] [(-e | --entire)] [(-i | --ignore-case)] [(-r | --regex)] [(-n | --index)] [(-q | --quiet)] [(-v | --invert)] PATTERN [STRING...] string pad [(-r | --right)] [(-c | --char) CHAR] [(-w | --width) INTEGER] [STRING...] string repeat [(-n | --count) COUNT] [(-m | --max) MAX] [(-N | --no-newline)] [(-q | --quiet)] [STRING...] string replace [(-a | --all)] [(-f | --filter)] [(-i | --ignore-case)] [(-r | --regex)] [(-q | --quiet)] PATTERN REPLACEMENT [STRING...] string split [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] SEP [STRING...] string split0 [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] [STRING...] string sub [(-s | --start) START] [(-l | --length) LENGTH] [(-q | --quiet)] [STRING...] string trim [(-l | --left)] [(-r | --right)] [(-c | --chars CHARS)] [(-q | --quiet)] [STRING...] string unescape [--style=xxx] [STRING...] string upper [(-q | --quiet)] [STRING...] Descriptionstring performs operations on strings.STRING arguments are taken from the command line unless standard input is connected to a pipe or a file, in which case they are read from standard input, one STRING per line. It is an error to supply STRING arguments on the command line and on standard input. Arguments beginning with - are normally interpreted as switches; -- causes the following arguments not to be treated as switches even if they begin with -. Switches and required arguments are recognized only on the command line. Most subcommands accept a -q or --quiet switch, which suppresses the usual output but exits with the documented status. In this case these commands will quit early, without reading all of the available input. The following subcommands are available. collect subcommandstring collect [(-N | --no-trim-newlines)] [STRING...] string collect collects its input into a single output argument, without splitting the output when used in a command substitution. This is useful when trying to collect multiline output from another command into a variable. Exit status: 0 if any output argument is non-empty, or 1 otherwise. If invoked with multiple arguments instead of input, string collect preserves each argument separately, where the number of output arguments is equal to the number of arguments given to string collect. Any trailing newlines on the input are trimmed, just as with "$(cmd)" substitution in sh. --no-trim-newlines can be used to disable this behavior, which may be useful when running a command such as set contents (cat filename | string collect -N). Examples>_ echo \"(echo one\ntwo\nthree | string collect)\" "one two three" >_ echo \"(echo one\ntwo\nthree | string collect -N)\" "one two three " escape and unescape subcommandsstring escape [(-n | --no-quoted)] [--style=xxx] [STRING...] string unescape [--style=xxx] [STRING...] string escape escapes each STRING in one of three ways. The first is --style=script. This is the default. It alters the string such that it can be passed back to eval to produce the original argument again. By default, all special characters are escaped, and quotes are used to simplify the output when possible. If -n or --no-quoted is given, the simplifying quoted format is not used. Exit status: 0 if at least one string was escaped, or 1 otherwise. --style=var ensures the string can be used as a variable name by hex encoding any non-alphanumeric characters. The string is first converted to UTF-8 before being encoded. --style=url ensures the string can be used as a URL by hex encoding any character which is not legal in a URL. The string is first converted to UTF-8 before being encoded. --style=regex escapes an input string for literal matching within a regex expression. The string is first converted to UTF-8 before being encoded. string unescape performs the inverse of the string escape command. If the string to be unescaped is not properly formatted it is ignored. For example, doing string unescape --style=var (string escape --style=var $str) will return the original string. There is no support for unescaping --style=regex. Examples>_ echo \x07 | string escape \cg >_ string escape --style=var 'a1 b2'\u6161 a1_20_b2_E6_85_A1_ join and join0 subcommandsstring join [(-q | --quiet)] SEP [STRING...] string join0 [(-q | --quiet)] [STRING...] string join joins its STRING arguments into a single string separated by SEP, which can be an empty string. Exit status: 0 if at least one join was performed, or 1 otherwise. string join0 joins its STRING arguments into a single string separated by the zero byte (NUL), and adds a trailing NUL. This is most useful in conjunction with tools that accept NUL-delimited input, such as sort -z. Exit status: 0 if at least one join was performed, or 1 otherwise. Because Unix uses NUL as the string terminator, passing the output of string join0 as an argument to a command (via a command substitution) won't actually work. Fish will pass the correct bytes along, but the command won't be able to tell where the argument ends. This is a limitation of Unix' argument passing. Examples>_ seq 3 | string join ... 1...2...3 # Give a list of NUL-separated filenames to du (this is a GNU extension) >_ string join0 file1 file2 file\nwith\nmultiple\nlines | du --files0-from=- # Just put the strings together without a separator >_ string join '' a b c abc length subcommandstring length [(-q | --quiet)] [STRING...] string length reports the length of each string argument in characters. Exit status: 0 if at least one non-empty STRING was given, or 1 otherwise. Examples>_ string length 'hello, world' 12 >_ set str foo >_ string length -q $str; echo $status 0 # Equivalent to test -n "$str" lower subcommandstring lower [(-q | --quiet)] [STRING...] string lower converts each string argument to lowercase. Exit status: 0 if at least one string was converted to lowercase, else 1. This means that in conjunction with the -q flag you can readily test whether a string is already lowercase. match subcommandstring match [(-a | --all)] [(-e | --entire)] [(-i | --ignore-case)] [(-r | --regex)] [(-n | --index)] [(-q | --quiet)] [(-v | --invert)] PATTERN [STRING...] string match tests each STRING against PATTERN and prints matching substrings. Only the first match for each STRING is reported unless -a or --all is given, in which case all matches are reported. If you specify the -e or --entire then each matching string is printed including any prefix or suffix not matched by the pattern (equivalent to grep without the -o flag). You can, obviously, achieve the same result by prepending and appending * or .* depending on whether or not you have specified the --regex flag. The --entire flag is simply a way to avoid having to complicate the pattern in that fashion and make the intent of the string match clearer. Without --entire and --regex, a PATTERN will need to match the entire STRING before it will be reported. Matching can be made case-insensitive with --ignore-case or -i. If --index or -n is given, each match is reported as a 1-based start position and a length. By default, PATTERN is interpreted as a glob pattern matched against each entire STRING argument. A glob pattern is only considered a valid match if it matches the entire STRING. 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 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. Exit status: 0 if at least one match was found, or 1 otherwise. Match Glob Examples>_ string match '?' a a >_ string match 'a*b' axxb axxb >_ string match -i 'a??B' Axxb Axxb >_ echo 'ok?' | string match '*\?' ok? # Note that only the second STRING will match here. >_ string match 'foo' 'foo1' 'foo' 'foo2' foo >_ string match -e 'foo' 'foo1' 'foo' 'foo2' foo1 foo foo2 >_ string match 'foo?' 'foo1' 'foo' 'foo2' foo1 foo2 Match Regex Examples>_ string match -r 'cat|dog|fish' 'nice dog' dog >_ string match -r -v "c.*[12]" {cat,dog}(seq 1 4) dog1 dog2 cat3 dog3 cat4 dog4 >_ string match -r '(\d\d?):(\d\d):(\d\d)' 2:34:56 2:34:56 2 34 56 >_ string match -r '^(\w{2,4})\1$' papa mud murmur papa pa murmur mur >_ string match -r -a -n at ratatat 2 2 4 2 6 2 >_ string match -r -i '0x[0-9a-f]{1,8}' 'int magic = 0xBadC0de;' 0xBadC0de >_ echo $version 3.1.2-1575-ga2ff32d90 >_ string match -rq '(?<major>\d+).(?<minor>\d+).(?<revision>\d+)' -- $version >_ echo "You are using fish $major!" You are using fish 3! >_ string match -raq ' *(?<sentence>[^.!?]+)(?<punctuation>[.!?])?' "hello, friend. goodbye" >_ printf "%s\n" -- $sentence hello, friend goodbye >_ printf "%s\n" -- $punctuation . >_ string match -rq '(?<word>hello)' 'hi' >_ count $word 0 pad subcommandstring pad [(-r | --right)] [(-c | --char) CHAR] [(-w | --width) INTEGER] [STRING...] string pad extends each STRING to the given width by adding CHAR to the left. If -r or --right is given, add the padding after a string. If -c or --char is given, pad with CHAR instead of whitespace. The output is padded to the maximum width of all input strings. If -w or --width is given, use at least that. >_ string pad -w 10 abc abcdef abc abcdef >_ string pad --right --char=🐟 "fish are pretty" "rich. " fish are pretty rich. 🐟🐟🐟🐟 >_ string pad -w$COLUMNS (date) # Prints the current time on the right edge of the screen. See Also
repeat subcommandstring repeat [(-n | --count) COUNT] [(-m | --max) MAX] [(-N | --no-newline)] [(-q | --quiet)] [STRING...] string repeat repeats the STRING -n or --count times. The -m or --max option will limit the number of outputted char (excluding the newline). This option can be used by itself or in conjunction with --count. If both --count and --max are present, max char will be outputed unless the final repeated string size is less than max, in that case, the string will repeat until count has been reached. Both --count and --max will accept a number greater than or equal to zero, in the case of zero, nothing will be outputed. If -N or --no-newline is given, the output won't contain a newline character at the end. Exit status: 0 if yielded string is not empty, 1 otherwise. ExamplesRepeat Examples>_ string repeat -n 2 'foo ' foo foo >_ echo foo | string repeat -n 2 foofoo >_ string repeat -n 2 -m 5 'foo' foofo >_ string repeat -m 5 'foo' foofo replace subcommandstring replace [(-a | --all)] [(-f | --filter)] [(-i | --ignore-case)] [(-r | --regex)] [(-q | --quiet)] PATTERN REPLACEMENT [STRING...] string replace is similar to string match but replaces non-overlapping matching substrings with a replacement string and prints the result. By default, PATTERN is treated as a literal substring to be matched. If -r or --regex is given, PATTERN is interpreted as a Perl-compatible regular expression, and REPLACEMENT can contain C-style escape sequences like \t as well as references to capturing groups by number or name as $n or ${n}. If you specify the -f or --filter flag then each input string is printed only if a replacement was done. This is useful where you would otherwise use this idiom: a_cmd | string match pattern | string replace pattern new_pattern. You can instead just write a_cmd | string replace --filter pattern new_pattern. Exit status: 0 if at least one replacement was performed, or 1 otherwise. Replace Literal Examples>_ string replace is was 'blue is my favorite' blue was my favorite >_ string replace 3rd last 1st 2nd 3rd 1st 2nd last >_ string replace -a ' ' _ 'spaces to underscores' spaces_to_underscores Replace Regex Examples>_ string replace -r -a '[^\d.]+' ' ' '0 one two 3.14 four 5x' 0 3.14 5 >_ string replace -r '(\w+)\s+(\w+)' '$2 $1 $$' 'left right' right left $ >_ string replace -r '\s*newline\s*' '\n' 'put a newline here' put a here split and split0 subcommandsstring split [(-f | --fields) FIELDS] [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] SEP [STRING...] string split0 [(-f | --fields) FIELDS] [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] [STRING...] string split splits each STRING on the separator SEP, which can be an empty string. If -m or --max is specified, at most MAX splits are done on each STRING. If -r or --right is given, splitting is performed right-to-left. This is useful in combination with -m or --max. With -n or --no-empty, empty results are excluded from consideration (e.g. hello\n\nworld would expand to two strings and not three). Exit status: 0 if at least one split was performed, or 1 otherwise. Use -f or --fields to print out specific fields. Unless --allow-empty is used, if a given field does not exist, then the command exits with status 1 and does not print anything. See also the --delimiter option of the read command. string split0 splits each STRING on the zero byte (NUL). Options are the same as string split except that no separator is given. split0 has the important property that its output is not further split when used in a command substitution, allowing for the command substitution to produce elements containing newlines. This is most useful when used with Unix tools that produce zero bytes, such as find -print0 or sort -z. See split0 examples below. Examples>_ string split . example.com example com >_ string split -r -m1 / /usr/local/bin/fish /usr/local/bin fish >_ string split '' abc a b c >_ string split --allow-empty -f1,3,5 '' abc a c NUL Delimited Examples>_ # Count files in a directory, without being confused by newlines. >_ count (find . -print0 | string split0) 42 >_ # Sort a list of elements which may contain newlines >_ set foo beta alpha\ngamma >_ set foo (string join0 $foo | sort -z | string split0) >_ string escape $foo[1] alpha\ngamma sub subcommandstring sub [(-s | --start) START] [(-e | --end) END] [(-l | --length) LENGTH] [(-q | --quiet)] [STRING...] string sub prints a substring of each string argument. The start/end of the substring can be specified with -s/-e or --start/--end followed by a 1-based index value. Positive index values are relative to the start of the string and negative index values are relative to the end of the string. The default start value is 1. The length of the substring can be specified with -l or --length. If the length or end is not specified, the substring continues to the end of each STRING. Exit status: 0 if at least one substring operation was performed, 1 otherwise. --length is mutually exclusive with --end. Examples>_ string sub --length 2 abcde ab >_ string sub -s 2 -l 2 abcde bc >_ string sub --start=-2 abcde de >_ string sub --end=3 abcde abc >_ string sub -e -1 abcde abcd >_ string sub -s 2 -e -1 abcde bcd >_ string sub -s -3 -e -2 abcde c trim subcommandstring trim [(-l | --left)] [(-r | --right)] [(-c | --chars CHARS)] [(-q | --quiet)] [STRING...] string trim removes leading and trailing whitespace from each STRING. If -l or --left is given, only leading whitespace is removed. If -r or --right is given, only trailing whitespace is trimmed. The -c or --chars switch causes the characters in CHARS to be removed instead of whitespace. Exit status: 0 if at least one character was trimmed, or 1 otherwise. Examples>_ string trim ' abc ' abc >_ string trim --right --chars=yz xyzzy zany x zan upper subcommandstring upper [(-q | --quiet)] [STRING...] string upper converts each string argument to uppercase. Exit status: 0 if at least one string was converted to uppercase, else 1. This means that in conjunction with the -q flag you can readily test whether a string is already uppercase. Regular ExpressionsBoth the match and replace subcommand support regular expressions when used with the -r or --regex option. The dialect is that of PCRE2.In general, special characters are special by default, so a+ matches one or more "a"s, while a\+ matches an "a" and then a "+". (a+) matches one or more "a"s in a capturing group ((?:XXXX) denotes a non-capturing group). For the replacement parameter of replace, $n refers to the n-th group of the match. In the match parameter, \n (e.g. \1) refers back to groups. Some features include repetitions:
Character classes, some of the more important:
Groups:
And some other things:
Comparison to other toolsMost operations string supports can also be done by external tools. Some of these include grep, sed and cut.If you are familiar with these, it is useful to know how string differs from them. In contrast to these classics, string reads input either from stdin or as arguments. string also does not deal with files, so it requires redirections to be used with them. In contrast to grep, string's match defaults to glob-mode, while replace defaults to literal matching. If set to regex-mode, they use PCRE regular expressions, which is comparable to grep's -P option. match defaults to printing just the match, which is like grep with -o (use --entire to enable grep-like behavior). Like sed's s/old/new/ command, string replace still prints strings that don't match. sed's -n in combination with a /p modifier or command is like string replace -f. string split somedelimiter is a replacement for tr somedelimiter \n. string-collect - join strings into oneSynopsisstring collect [(-N | --no-trim-newlines)] [STRING...] Descriptionstring collect collects its input into a single output argument, without splitting the output when used in a command substitution. This is useful when trying to collect multiline output from another command into a variable. Exit status: 0 if any output argument is non-empty, or 1 otherwise.If invoked with multiple arguments instead of input, string collect preserves each argument separately, where the number of output arguments is equal to the number of arguments given to string collect. Any trailing newlines on the input are trimmed, just as with "$(cmd)" substitution in sh. --no-trim-newlines can be used to disable this behavior, which may be useful when running a command such as set contents (cat filename | string collect -N). Examples>_ echo \"(echo one\ntwo\nthree | string collect)\" "one two three" >_ echo \"(echo one\ntwo\nthree | string collect -N)\" "one two three " string-escape - escape special charactersSynopsisstring escape [(-n | --no-quoted)] [--style=xxx] [STRING...] string unescape [--style=xxx] [STRING...] Descriptionstring escape escapes each STRING in one of three ways. The first is --style=script. This is the default. It alters the string such that it can be passed back to eval to produce the original argument again. By default, all special characters are escaped, and quotes are used to simplify the output when possible. If -n or --no-quoted is given, the simplifying quoted format is not used. Exit status: 0 if at least one string was escaped, or 1 otherwise.--style=var ensures the string can be used as a variable name by hex encoding any non-alphanumeric characters. The string is first converted to UTF-8 before being encoded. --style=url ensures the string can be used as a URL by hex encoding any character which is not legal in a URL. The string is first converted to UTF-8 before being encoded. --style=regex escapes an input string for literal matching within a regex expression. The string is first converted to UTF-8 before being encoded. string unescape performs the inverse of the string escape command. If the string to be unescaped is not properly formatted it is ignored. For example, doing string unescape --style=var (string escape --style=var $str) will return the original string. There is no support for unescaping --style=regex. Examples>_ echo \x07 | string escape \cg >_ string escape --style=var 'a1 b2'\u6161 a1_20_b2_E6_85_A1_ string-join - join strings with delimiterSynopsisstring join [(-q | --quiet)] SEP [STRING...] string join0 [(-q | --quiet)] [STRING...] Descriptionstring join joins its STRING arguments into a single string separated by SEP, which can be an empty string. Exit status: 0 if at least one join was performed, or 1 otherwise.string join0 joins its STRING arguments into a single string separated by the zero byte (NUL), and adds a trailing NUL. This is most useful in conjunction with tools that accept NUL-delimited input, such as sort -z. Exit status: 0 if at least one join was performed, or 1 otherwise. Because Unix uses NUL as the string terminator, passing the output of string join0 as an argument to a command (via a command substitution) won't actually work. Fish will pass the correct bytes along, but the command won't be able to tell where the argument ends. This is a limitation of Unix' argument passing. Examples>_ seq 3 | string join ... 1...2...3 # Give a list of NUL-separated filenames to du (this is a GNU extension) >_ string join0 file1 file2 file\nwith\nmultiple\nlines | du --files0-from=- # Just put the strings together without a separator >_ string join '' a b c abc string-join0 - join strings with zero bytesSynopsisstring join [(-q | --quiet)] SEP [STRING...] string join0 [(-q | --quiet)] [STRING...] Descriptionstring join joins its STRING arguments into a single string separated by SEP, which can be an empty string. Exit status: 0 if at least one join was performed, or 1 otherwise.string join0 joins its STRING arguments into a single string separated by the zero byte (NUL), and adds a trailing NUL. This is most useful in conjunction with tools that accept NUL-delimited input, such as sort -z. Exit status: 0 if at least one join was performed, or 1 otherwise. Because Unix uses NUL as the string terminator, passing the output of string join0 as an argument to a command (via a command substitution) won't actually work. Fish will pass the correct bytes along, but the command won't be able to tell where the argument ends. This is a limitation of Unix' argument passing. Examples>_ seq 3 | string join ... 1...2...3 # Give a list of NUL-separated filenames to du (this is a GNU extension) >_ string join0 file1 file2 file\nwith\nmultiple\nlines | du --files0-from=- # Just put the strings together without a separator >_ string join '' a b c abc string-length - print string lengthsSynopsisstring length [(-q | --quiet)] [STRING...] Descriptionstring length reports the length of each string argument in characters. Exit status: 0 if at least one non-empty STRING was given, or 1 otherwise.Examples>_ string length 'hello, world' 12 >_ set str foo >_ string length -q $str; echo $status 0 # Equivalent to test -n "$str" string-lower - convert strings to lowercaseSynopsisstring lower [(-q | --quiet)] [STRING...] Descriptionstring lower converts each string argument to lowercase. Exit status: 0 if at least one string was converted to lowercase, else 1. This means that in conjunction with the -q flag you can readily test whether a string is already lowercase.string-match - match substringsSynopsisstring match [(-a | --all)] [(-e | --entire)] [(-i | --ignore-case)] [(-r | --regex)] [(-n | --index)] [(-q | --quiet)] [(-v | --invert)] PATTERN [STRING...] Descriptionstring match tests each STRING against PATTERN and prints matching substrings. Only the first match for each STRING is reported unless -a or --all is given, in which case all matches are reported.If you specify the -e or --entire then each matching string is printed including any prefix or suffix not matched by the pattern (equivalent to grep without the -o flag). You can, obviously, achieve the same result by prepending and appending * or .* depending on whether or not you have specified the --regex flag. The --entire flag is simply a way to avoid having to complicate the pattern in that fashion and make the intent of the string match clearer. Without --entire and --regex, a PATTERN will need to match the entire STRING before it will be reported. Matching can be made case-insensitive with --ignore-case or -i. If --index or -n is given, each match is reported as a 1-based start position and a length. By default, PATTERN is interpreted as a glob pattern matched against each entire STRING argument. A glob pattern is only considered a valid match if it matches the entire STRING. 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 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. Exit status: 0 if at least one match was found, or 1 otherwise. ExamplesMatch Glob Examples>_ string match '?' a a >_ string match 'a*b' axxb axxb >_ string match -i 'a??B' Axxb Axxb >_ echo 'ok?' | string match '*\?' ok? # Note that only the second STRING will match here. >_ string match 'foo' 'foo1' 'foo' 'foo2' foo >_ string match -e 'foo' 'foo1' 'foo' 'foo2' foo1 foo foo2 >_ string match 'foo?' 'foo1' 'foo' 'foo2' foo1 foo2 Match Regex Examples>_ string match -r 'cat|dog|fish' 'nice dog' dog >_ string match -r -v "c.*[12]" {cat,dog}(seq 1 4) dog1 dog2 cat3 dog3 cat4 dog4 >_ string match -r '(\d\d?):(\d\d):(\d\d)' 2:34:56 2:34:56 2 34 56 >_ string match -r '^(\w{2,4})\1$' papa mud murmur papa pa murmur mur >_ string match -r -a -n at ratatat 2 2 4 2 6 2 >_ string match -r -i '0x[0-9a-f]{1,8}' 'int magic = 0xBadC0de;' 0xBadC0de >_ echo $version 3.1.2-1575-ga2ff32d90 >_ string match -rq '(?<major>\d+).(?<minor>\d+).(?<revision>\d+)' -- $version >_ echo "You are using fish $major!" You are using fish 3! >_ string match -raq ' *(?<sentence>[^.!?]+)(?<punctuation>[.!?])?' "hello, friend. goodbye" >_ printf "%s\n" -- $sentence hello, friend goodbye >_ printf "%s\n" -- $punctuation . >_ string match -rq '(?<word>hello)' 'hi' >_ count $word 0 string-pad - pad strings to a fixed widthSynopsisstring pad [(-r | --right)] [(-c | --char) CHAR] [(-w | --width) INTEGER] [STRING...] Descriptionstring pad extends each STRING to the given width by adding CHAR to the left.If -r or --right is given, add the padding after a string. If -c or --char is given, pad with CHAR instead of whitespace. The output is padded to the maximum width of all input strings. If -w or --width is given, use at least that. Examples>_ string pad -w 10 abc abcdef abc abcdef >_ string pad --right --char=🐟 "fish are pretty" "rich. " fish are pretty rich. 🐟🐟🐟🐟 >_ string pad -w$COLUMNS (date) # Prints the current time on the right edge of the screen. See Also
string-repeat - multiply a stringSynopsisstring repeat [(-n | --count) COUNT] [(-m | --max) MAX] [(-N | --no-newline)] [(-q | --quiet)] [STRING...] Descriptionstring repeat repeats the STRING -n or --count times. The -m or --max option will limit the number of outputted char (excluding the newline). This option can be used by itself or in conjunction with --count. If both --count and --max are present, max char will be outputed unless the final repeated string size is less than max, in that case, the string will repeat until count has been reached. Both --count and --max will accept a number greater than or equal to zero, in the case of zero, nothing will be outputed. If -N or --no-newline is given, the output won't contain a newline character at the end. Exit status: 0 if yielded string is not empty, 1 otherwise.ExamplesRepeat Examples>_ string repeat -n 2 'foo ' foo foo >_ echo foo | string repeat -n 2 foofoo >_ string repeat -n 2 -m 5 'foo' foofo >_ string repeat -m 5 'foo' foofo string-replace - replace substringsSynopsisstring replace [(-a | --all)] [(-f | --filter)] [(-i | --ignore-case)] [(-r | --regex)] [(-q | --quiet)] PATTERN REPLACEMENT [STRING...] Descriptionstring replace is similar to string match but replaces non-overlapping matching substrings with a replacement string and prints the result. By default, PATTERN is treated as a literal substring to be matched.If -r or --regex is given, PATTERN is interpreted as a Perl-compatible regular expression, and REPLACEMENT can contain C-style escape sequences like \t as well as references to capturing groups by number or name as $n or ${n}. If you specify the -f or --filter flag then each input string is printed only if a replacement was done. This is useful where you would otherwise use this idiom: a_cmd | string match pattern | string replace pattern new_pattern. You can instead just write a_cmd | string replace --filter pattern new_pattern. Exit status: 0 if at least one replacement was performed, or 1 otherwise. ExamplesReplace Literal Examples>_ string replace is was 'blue is my favorite' blue was my favorite >_ string replace 3rd last 1st 2nd 3rd 1st 2nd last >_ string replace -a ' ' _ 'spaces to underscores' spaces_to_underscores Replace Regex Examples>_ string replace -r -a '[^\d.]+' ' ' '0 one two 3.14 four 5x' 0 3.14 5 >_ string replace -r '(\w+)\s+(\w+)' '$2 $1 $$' 'left right' right left $ >_ string replace -r '\s*newline\s*' '\n' 'put a newline here' put a here string-split - split strings by delimiterSynopsisstring split [(-f | --fields) FIELDS] [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] SEP [STRING...] string split0 [(-f | --fields) FIELDS] [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] [STRING...] Descriptionstring split splits each STRING on the separator SEP, which can be an empty string. If -m or --max is specified, at most MAX splits are done on each STRING. If -r or --right is given, splitting is performed right-to-left. This is useful in combination with -m or --max. With -n or --no-empty, empty results are excluded from consideration (e.g. hello\n\nworld would expand to two strings and not three). Exit status: 0 if at least one split was performed, or 1 otherwise.Use -f or --fields to print out specific fields. Unless --allow-empty is used, if a given field does not exist, then the command exits with status 1 and does not print anything. See also the --delimiter option of the read command. string split0 splits each STRING on the zero byte (NUL). Options are the same as string split except that no separator is given. split0 has the important property that its output is not further split when used in a command substitution, allowing for the command substitution to produce elements containing newlines. This is most useful when used with Unix tools that produce zero bytes, such as find -print0 or sort -z. See split0 examples below. Examples>_ string split . example.com example com >_ string split -r -m1 / /usr/local/bin/fish /usr/local/bin fish >_ string split '' abc a b c >_ string split --allow-empty -f1,3,5 '' abc a c NUL Delimited Examples>_ # Count files in a directory, without being confused by newlines. >_ count (find . -print0 | string split0) 42 >_ # Sort a list of elements which may contain newlines >_ set foo beta alpha\ngamma >_ set foo (string join0 $foo | sort -z | string split0) >_ string escape $foo[1] alpha\ngamma string-split0 - split on zero bytesSynopsisstring split [(-f | --fields) FIELDS] [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] SEP [STRING...] string split0 [(-f | --fields) FIELDS] [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] [STRING...] Descriptionstring split splits each STRING on the separator SEP, which can be an empty string. If -m or --max is specified, at most MAX splits are done on each STRING. If -r or --right is given, splitting is performed right-to-left. This is useful in combination with -m or --max. With -n or --no-empty, empty results are excluded from consideration (e.g. hello\n\nworld would expand to two strings and not three). Exit status: 0 if at least one split was performed, or 1 otherwise.Use -f or --fields to print out specific fields. Unless --allow-empty is used, if a given field does not exist, then the command exits with status 1 and does not print anything. See also the --delimiter option of the read command. string split0 splits each STRING on the zero byte (NUL). Options are the same as string split except that no separator is given. split0 has the important property that its output is not further split when used in a command substitution, allowing for the command substitution to produce elements containing newlines. This is most useful when used with Unix tools that produce zero bytes, such as find -print0 or sort -z. See split0 examples below. Examples>_ string split . example.com example com >_ string split -r -m1 / /usr/local/bin/fish /usr/local/bin fish >_ string split '' abc a b c >_ string split --allow-empty -f1,3,5 '' abc a c NUL Delimited Examples>_ # Count files in a directory, without being confused by newlines. >_ count (find . -print0 | string split0) 42 >_ # Sort a list of elements which may contain newlines >_ set foo beta alpha\ngamma >_ set foo (string join0 $foo | sort -z | string split0) >_ string escape $foo[1] alpha\ngamma string-sub - extract substringsSynopsisstring sub [(-s | --start) START] [(-e | --end) END] [(-l | --length) LENGTH] [(-q | --quiet)] [STRING...] Descriptionstring sub prints a substring of each string argument. The start/end of the substring can be specified with -s/-e or --start/--end followed by a 1-based index value. Positive index values are relative to the start of the string and negative index values are relative to the end of the string. The default start value is 1. The length of the substring can be specified with -l or --length. If the length or end is not specified, the substring continues to the end of each STRING. Exit status: 0 if at least one substring operation was performed, 1 otherwise. --length is mutually exclusive with --end.Examples>_ string sub --length 2 abcde ab >_ string sub -s 2 -l 2 abcde bc >_ string sub --start=-2 abcde de >_ string sub --end=3 abcde abc >_ string sub -e -1 abcde abcd >_ string sub -s 2 -e -1 abcde bcd >_ string sub -s -3 -e -2 abcde c string-trim - remove trailing whitespaceSynopsisstring trim [(-l | --left)] [(-r | --right)] [(-c | --chars CHARS)] [(-q | --quiet)] [STRING...] Descriptionstring trim removes leading and trailing whitespace from each STRING. If -l or --left is given, only leading whitespace is removed. If -r or --right is given, only trailing whitespace is trimmed. The -c or --chars switch causes the characters in CHARS to be removed instead of whitespace. Exit status: 0 if at least one character was trimmed, or 1 otherwise.Examples>_ string trim ' abc ' abc >_ string trim --right --chars=yz xyzzy zany x zan string-unescape - expand escape sequencesSynopsisstring escape [(-n | --no-quoted)] [--style=xxx] [STRING...] string unescape [--style=xxx] [STRING...] Descriptionstring escape escapes each STRING in one of three ways. The first is --style=script. This is the default. It alters the string such that it can be passed back to eval to produce the original argument again. By default, all special characters are escaped, and quotes are used to simplify the output when possible. If -n or --no-quoted is given, the simplifying quoted format is not used. Exit status: 0 if at least one string was escaped, or 1 otherwise.--style=var ensures the string can be used as a variable name by hex encoding any non-alphanumeric characters. The string is first converted to UTF-8 before being encoded. --style=url ensures the string can be used as a URL by hex encoding any character which is not legal in a URL. The string is first converted to UTF-8 before being encoded. --style=regex escapes an input string for literal matching within a regex expression. The string is first converted to UTF-8 before being encoded. string unescape performs the inverse of the string escape command. If the string to be unescaped is not properly formatted it is ignored. For example, doing string unescape --style=var (string escape --style=var $str) will return the original string. There is no support for unescaping --style=regex. Examples>_ echo \x07 | string escape \cg >_ string escape --style=var 'a1 b2'\u6161 a1_20_b2_E6_85_A1_ string-upper - convert strings to uppercaseSynopsisstring upper [(-q | --quiet)] [STRING...] Descriptionstring upper converts each string argument to uppercase. Exit status: 0 if at least one string was converted to uppercase, else 1. This means that in conjunction with the -q flag you can readily test whether a string is already uppercase.suspend - suspend the current shellSynopsissuspend [--force] Descriptionsuspend suspends execution of the current shell by sending it a SIGTSTP signal, returning to the controlling process. It can be resumed later by sending it a SIGCONT. In order to prevent suspending a shell that doesn't have a controlling process, it will not suspend the shell if it is a login shell. This requirement is bypassed if the --force option is given or the shell is not interactive.switch - conditionally execute a block of commandsSynopsisswitch VALUE; [case [WILDCARD...]; [COMMANDS...]; ...] end Descriptionswitch performs one of several blocks of commands, depending on whether a specified value equals one of several wildcarded values. case is used together with the switch statement in order to determine which block should be executed.Each case command is given one or more parameters. The first case command with a parameter that matches the string specified in the switch command will be evaluated. case parameters may contain wildcards. These need to be escaped or quoted in order to avoid regular wildcard expansion using filenames. Note that fish does not fall through on case statements. Only the first matching case is executed. Note that command substitutions in a case statement will be evaluated even if its body is not taken. All substitutions, including command substitutions, must be performed before the value can be compared against the parameter. ExampleIf the variable $animal contains the name of an animal, the following code would attempt to classify it:switch $animal case cat echo evil case wolf dog human moose dolphin whale echo mammal case duck goose albatross echo bird case shark trout stingray echo fish case '*' echo I have no idea what a $animal is end If the above code was run with $animal set to whale, the output would be mammal. test - perform tests on files and textSynopsistest [EXPRESSION] [ [EXPRESSION] ] DescriptionTests the expression given and sets the exit status to 0 if true, and 1 if false. An expression is made up of one or more operators and their arguments.The first form (test) is preferred. For compatibility with other shells, the second form is available: a matching pair of square brackets ([ [EXPRESSION ] ]). This test is mostly POSIX-compatible. When using a variable as an argument for a test operator you should almost always enclose it in double-quotes. There are only two situations it is safe to omit the quote marks. The first is when the argument is a literal string with no whitespace or other characters special to the shell (e.g., semicolon). For example, test -b /my/file. The second is using a variable that expands to exactly one element including if that element is the empty string (e.g., set x ''). If the variable is not set, set but with no value, or set to more than one value you must enclose it in double-quotes. For example, test "$x" = "$y". Since it is always safe to enclose variables in double-quotes when used as test arguments that is the recommended practice. Operators for files and directories
Operators for text strings
Operators to compare and examine numbers
Both integers and floating point numbers are supported. Operators to combine expressions
Expressions can be inverted using the ! operator:
Expressions can be grouped using parentheses.
Note that parentheses will usually require escaping with
\( to avoid being interpreted as a command substitution.
ExamplesIf the /tmp directory exists, copy the /etc/motd file to it:if test -d /tmp cp /etc/motd /tmp/motd end If the variable MANPATH is defined and not empty, print the contents. (If MANPATH is not defined, then it will expand to zero arguments, unless quoted.) if test -n "$MANPATH" echo $MANPATH end Parentheses and the -o and -a operators can be combined to produce more complicated expressions. In this example, success is printed if there is a /foo or /bar file as well as a /baz or /bat file. if test \( -f /foo -o -f /bar \) -a \( -f /baz -o -f /bat \) echo Success. end. Numerical comparisons will simply fail if one of the operands is not a number: if test 42 -eq "The answer to life, the universe and everything" echo So long and thanks for all the fish # will not be executed end A common comparison is with $status: if test $status -eq 0 echo "Previous command succeeded" end The previous test can likewise be inverted: if test ! $status -eq 0 echo "Previous command failed" end which is logically equivalent to the following: if test $status -ne 0 echo "Previous command failed" end Standardstest implements a subset of the IEEE Std 1003.1-2008 (POSIX.1) standard. The following exceptions apply:
In cases such as this, one can use command
test to explicitly use the system's standalone test rather than
this builtin test.
time - measure how long a command or block takesSynopsistime COMMAND Descriptiontime causes fish to measure how long a command takes and print the results afterwards. The command can be a simple fish command or a block. The results can not currently be redirected.For checking timing after a command has completed, check $CMD_DURATION. Your system most likely also has a time command. To use that use something like command time, as in command time sleep 10. Because it's not inside fish, it won't have access to fish functions and won't be able to time blocks and such. Example(for obvious reasons exact results will vary on your system)>_ time sleep 1s ________________________________________________________ Executed in 1,01 secs fish external usr time 2,32 millis 0,00 micros 2,32 millis sys time 0,88 millis 877,00 micros 0,00 millis >_ time for i in 1 2 3; sleep 1s; end ________________________________________________________ Executed in 3,01 secs fish external usr time 9,16 millis 2,94 millis 6,23 millis sys time 0,23 millis 0,00 millis 0,23 millis Inline variable assignments need to follow the time keyword: >_ time a_moment=1.5m sleep $a_moment ________________________________________________________ Executed in 90.00 secs fish external usr time 4.62 millis 4.62 millis 0.00 millis sys time 2.35 millis 0.41 millis 1.95 millis trap - perform an action when the shell receives a signalSynopsistrap [OPTIONS] [[ARG] REASON ... ] Descriptiontrap is a wrapper around the fish event delivery framework. It exists for backwards compatibility with POSIX shells. For other uses, it is recommended to define an event handler.The following parameters are available:
If ARG and REASON are both specified, ARG is the command to be executed when the event specified by REASON occurs (e.g., the signal is delivered). If ARG is absent (and there is a single REASON) or -, each specified signal is reset to its original disposition (the value it had upon entrance to the shell). If ARG is the null string the signal specified by each REASON is ignored by the shell and by the commands it invokes. If ARG is not present and -p has been supplied, then the trap commands associated with each REASON are displayed. If no arguments are supplied or if only -p is given, trap prints the list of commands associated with each signal. Signal names are case insensitive and the SIG prefix is optional. The exit status is 1 if any REASON is invalid; otherwise trap returns 0. Exampletrap "status --print-stack-trace" SIGUSR1 # Prints a stack trace each time the SIGUSR1 signal is sent to the shell. true - return a successful resultSynopsistrue Descriptiontrue sets the exit status to 0.See Also
type - indicate how a command would be interpretedSynopsistype [OPTIONS] NAME [NAME ...] DescriptionWith no options, type indicates how each NAME would be interpreted if used as a command name.The following options are available:
The -q, -p, -t and -P flags (and their long flag aliases) are mutually exclusive. Only one can be specified at a time. Example>_ type fg fg is a builtin ulimit - set or get resource usage limitsSynopsisulimit [OPTIONS] [LIMIT] Descriptionulimit sets or outputs the resource usage limits of the shell and any processes spawned by it. If a new limit value is omitted, the current value of the limit of the resource is printed; otherwise, the specified limit is set to the new value.Use one of the following switches to specify which resource limit to set or report:
Note that not all these limits are available in all operating systems. The value of limit can be a number in the unit specified for the resource or one of the special values hard, soft, or unlimited, which stand for the current hard limit, the current soft limit, and no limit, respectively. If limit is given, it is the new value of the specified resource. If no option is given, then -f is assumed. Values are in kilobytes, except for -t, which is in seconds and -n and -u, which are unscaled values. The exit status is 0 unless an invalid option or argument is supplied, or an error occurs while setting a new limit. ulimit also accepts the following switches that determine what type of limit to set:
A hard limit can only be decreased. Once it is set it cannot be increased; a soft limit may be increased up to the value of the hard limit. If neither -H nor -S is specified, both the soft and hard limits are updated when assigning a new limit value, and the soft limit is used when reporting the current value. The following additional options are also understood by ulimit:
The fish implementation of ulimit should behave identically to the implementation in bash, except for these differences:
Exampleulimit -Hs 64 sets the hard stack size limit to 64 kB.umask - set or get the file creation mode maskSynopsisumask [OPTIONS] [MASK] Descriptionumask displays and manipulates the "umask", or file creation mode mask, which is used to restrict the default access to files.The umask may be expressed either as an octal number, which represents the rights that will be removed by default, or symbolically, which represents the only rights that will be granted by default. Access rights are explained in the manual page for the chmod(1) program. With no parameters, the current file creation mode mask is printed as an octal number.
If a numeric mask is specified as a parameter, the current shell's umask will be set to that value, and the rights specified by that mask will be removed from new files and directories by default. If a symbolic mask is specified, the desired permission bits, and not the inverse, should be specified. A symbolic mask is a comma separated list of rights. Each right consists of three parts:
If the first and second parts are skipped, they are assumed to be a and =, respectively. As an example, r,u+w means all users should have read access and the file owner should also have write access. Note that symbolic masks currently do not work as intended. Exampleumask 177 or umask u=rw sets the file creation mask to read and write for the owner and no permissions at all for any other users.vared - interactively edit the value of an environment variableSynopsisvared VARIABLE_NAME Descriptionvared is used to interactively edit the value of an environment variable. Array variables as a whole can not be edited using vared, but individual list elements can.Examplevared PATH[3] edits the third element of the PATH listwait - wait for jobs to completeSynopsiswait [-n | --any] [PID | PROCESS_NAME] ... Descriptionwait waits for child jobs to complete.
Examplesleep 10 & wait $last_pid spawns sleep in the background, and then waits until it finishes. for i in (seq 1 5); sleep 10 &; end wait spawns five jobs in the background, and then waits until all of them finishes. for i in (seq 1 5); sleep 10 &; end hoge & wait sleep spawns five jobs and hoge in the background, and then waits until all sleeps finish, and doesn't wait for hoge finishing. while - perform a command multiple timesSynopsiswhile CONDITION; COMMANDS...; end Descriptionwhile repeatedly executes CONDITION, and if the exit status is 0, then executes COMMANDS.The exit status of the while loop is the exit status of the last iteration of the COMMANDS executed, or 0 if none were executed. (This matches other shells and is POSIX-compatible.) You can use and or or for complex conditions. Even more complex control can be achieved with while true containing a break. Examplewhile test -f foo.txt; or test -f bar.txt ; echo file exists; sleep 10; end # outputs 'file exists' at 10 second intervals, # as long as the file foo.txt or bar.txt exists. Fish for bash usersThis is to give you a quick overview if you come from bash (or to a lesser extent other shells like zsh or ksh) and want to know how fish differs. Fish is intentionally not POSIX-compatible and as such some of the things you are used to work differently.Many things are similar - they both fundamentally expand commandlines to execute commands, have pipes, redirections, variables, globs, use command output in various ways. This document is there to quickly show you the differences. Command substitutionsFish spells command substitutions as (command) instead of $(command) (or `command`).In addition, it only splits them on newlines instead of $IFS. If you want to split on something else, use string split, string split0 or string collect. If those are used as the last command in a command substitution the splits they create are carried over. So: for i in (find . -print0 | string split0) will correctly handle all possible filenames. VariablesFish sets and erases variables with set instead of VAR=VAL and declare and unset and export. set takes options to determine the scope and exportedness of a variable:# Define $PAGER global and exported, so this is like ``export PAGER=less`` set -gx PAGER less # Define $alocalvariable only locally - like ``local alocalvariable=foo`` set -l alocalvariable foo or to erase variables: set -e PAGER VAR=VAL statements are available as environment overrides: PAGER=cat git log Fish does not perform word splitting. Once a variable has been set to a value, that value stays as it is, so double-quoting variable expansions isn't the necessity it is in bash. [1] For instance, here's bash > foo="bar baz" > printf '"%s"\n' $foo # will print two lines, because we didn't double-quote, so the variable is split "bar" "baz" And here is fish: > set foo "bar baz" > printf '"%s"\n' $foo # foo was set as one element, so it will be passed as one element, so this is one line "bar baz" All variables are "arrays" (we use the term "lists"), and expanding a variable expands to all its elements, with each element as its own argument (like bash's "${var[@]}": > set var "foo bar" banana > printf %s\n $var foo bar banana Specific elements of a list can be selected: echo $list[5..7]
Wildcards (globs)Fish only supports the * and ** glob (and the deprecated ? glob). If a glob doesn't match it fails the command (like with bash's failglob) unless the command is for, set or count or the glob is used with an environment override (VAR=* command), in which case it expands to nothing (like with bash's nullglob option).Globbing doesn't happen on expanded variables, so: set foo "*" echo $foo will not match any files. There are no options to control globbing so it always behaves like that. QuotingFish has two quoting styles: "" and ''. Variables are expanded in double-quotes, nothing is expanded in single-quotes.There is no $'', instead the sequences that would transform are transformed when unquoted: > echo a\nb a b String manipulationFish does not have ${foo%bar}, ${foo#bar} and ${foo/bar/baz}. Instead string manipulation is done by the string builtin.Special variablesSome bash variables and their closest fish equivalent:
Process substitutionInstead of <(command) fish uses (command | psub). There is no equivalent to >(command).Note that both of these are bashisms, and most things can easily be expressed without. E.g. instead of: source (command | psub) just use: command | source as fish's source can read from stdin. HeredocsFish does not have <<EOF "heredocs". Instead of:cat <<EOF some string some more string EOF use: printf %s\n "some string" "some more string" or: echo "some string some more string" Quotes are followed across newlines. Test (test, [, [[)Fish has a POSIX-compatible test or [ builtin. There is no [[ and test does not accept == as a synonym for =. It can compare floating point numbers, however.set -q can be used to determine if a variable exists or has a certain number of elements (set -q foo[2]). Arithmetic ExpansionFish does not have $((i+1)) arithmetic expansion, computation is handled by math:math $i + 1 It can handle floating point numbers: > math 5 / 2 2.5 PromptsFish does not use the $PS1, $PS2 and so on variables. Instead the prompt is the output of the fish_prompt function, plus the fish_mode_prompt function if vi-mode is enabled and the fish_right_prompt function for the right prompt.As an example, here's a relatively simple bash prompt: # <$HOSTNAME> <$PWD in blue> <Prompt Sign in Yellow> <Rest in default light white> export PS1='\h\[\e[1;34m\]\w\[\e[m\] \[\e[1;32m\]\$\[\e[m\] ' and a rough fish equivalent: function fish_prompt set -l prompt_symbol '$' fish_is_root_user; and set prompt_symbol '#' echo -s $hostname (set_color blue) (prompt_pwd) \ (set_color yellow) $prompt_symbol (set_color normal) end This shows a few differences:
The default prompt is reasonably full-featured and its code can be read via type fish_prompt. Fish does not have $PS2 for continuation lines, instead it leaves the lines indented to show that the commandline isn't complete yet. Blocks and loopsFish's blocking constructs look a little different. They all start with a word, end in end and don't have a second starting word:for i in 1 2 3; do echo $i done # becomes for i in 1 2 3 echo $i end while true; do echo Weeee done # becomes while true echo Weeeeeee end { echo Hello } # becomes begin echo Hello end if true; then echo Yes I am true else echo "How is true not true?" fi # becomes if true echo Yes I am true else echo "How is true not true?" end foo() { echo foo } # becomes function foo echo foo end # (note that bash specifically allows the word "function" as an extension, but POSIX only specifies the form without, so it's more compatible to just use the form without) Fish does not have an until. Use while not or while !. SubshellsBash has a feature called "subshells", where it will start another shell process for certain things. That shell will then be independent and e.g. any changes it makes to variables won't be visible in the main shell.This includes things like: # A list of commands in `()` parentheses (foo; bar) | baz # Both sides of a pipe foo | while read -r bar; do # This variable will not be visible outside of the while loop. VAR=VAL # This background process will not be, either baz & done () subshells are often confused with {} grouping, which does not use a subshell. When you just need to group, you can use begin; end in fish: (foo; bar) | baz # when it should really have been: { foo; bar } | baz # becomes begin; foo; bar; end | baz The pipe will simply be run in the same process, so while read loops can set variables outside: foo | while read bar set -g VAR VAL baz & end echo $VAR # will print VAL jobs # will show "baz" Subshells are also frequently confused with command substitutions, which bash writes as `command` or $(command) and fish writes as (command). Bash also uses subshells to implement them. The isolation can usually be achieved by just scoping variables (with set -l), but if you really do need to run your code in a new shell environment you can always use fish -c 'your code here' to do so explicitly. Builtins and other commandsBy now it has become apparent that fish puts much more of a focus on its builtins and external commands rather than its syntax. So here are some helpful builtins and their rough equivalent in bash:
TutorialWhy fish?Fish is a fully-equipped command line shell (like bash or zsh) that is smart and user-friendly. Fish supports powerful features like syntax highlighting, autosuggestions, and tab completions that just work, with nothing to learn or configure.If you want to make your command line more productive, more useful, and more fun, without learning a bunch of arcane syntax and configuration options, then fish might be just what you're looking for! Getting startedOnce installed, just type in fish into your current shell to try it out!You will be greeted by the standard fish prompt, which means you are all set up and can start using fish: > fish Welcome to fish, the friendly interactive shell Type help for instructions on how to use fish you@hostname ~> This prompt that you see above is the fish default prompt: it shows your username, hostname, and working directory. - to change this prompt see how to change your prompt - to switch to fish permanently see switch your default shell to fish. From now on, we'll pretend your prompt is just a > to save space. Learning fishThis tutorial assumes a basic understanding of command line shells and Unix commands, and that you have a working copy of fish.If you have a strong understanding of other shells, and want to know what fish does differently, search for the magic phrase unlike other shells, which is used to call out important differences. Or, if you want a quick overview over the differences to other shells like Bash, see Fish For Bash Users. Running CommandsFish runs commands like other shells: you type a command, followed by its arguments. Spaces are separators:> echo hello world hello world This runs the command echo with the arguments hello and world. In this case that's the same as one argument hello world, but in many cases it's not. If you need to pass an argument that includes a space, you can escape with a backslash, or quote it using single or double quotes: > mkdir My\ Files # Makes a directory called "My Files", with a space in the name > cp ~/Some\ File 'My Files' # Copies a file called "Some File" in the home directory to "My Files" > ls "My Files" Some File Getting HelpRun help to open fish's help in a web browser, and man with the page (like fish-language) to open it in a man page. You can also ask for help with a specific command, for example, help set to open in a web browser, or man set to see it in the terminal.> man set set - handle shell variables Synopsis... Syntax HighlightingYou'll quickly notice that fish performs syntax highlighting as you type. Invalid commands are colored red by default:> /bin/mkd A command may be invalid because it does not exist, or refers to a file that you cannot execute. When the command becomes valid, it is shown in a different color: > /bin/mkdir Valid file paths are underlined as you type them: > cat ~/somefi This tells you that there exists a file that starts with somefi, which is useful feedback as you type. These colors, and many more, can be changed by running fish_config, or by modifying color variables directly. WildcardsFish supports the familiar wildcard *. To list all JPEG files:> ls *.jpg lena.jpg meena.jpg santa maria.jpg You can include multiple wildcards: > ls l*.p* lena.png lesson.pdf Especially powerful is the recursive wildcard ** which searches directories recursively: > ls /var/**.log /var/log/system.log /var/run/sntp.log If that directory traversal is taking a long time, you can Control+C out of it. For more, see Wildcards. Pipes and RedirectionsYou can pipe between commands with the usual vertical bar:> echo hello world | wc 1 2 12 stdin and stdout can be redirected via the familiar < and >. stderr is redirected with a 2>. > grep fish < /etc/shells > ~/output.txt 2> ~/errors.txt To redirect stdout and stderr into one file, you need to first redirect stdout, and then stderr into stdout: > make > make_output.txt 2>&1 For more, see Input and output redirections and Pipes. AutosuggestionsAs you type fish will suggest commands to the right of the cursor, in gray. For example:> /bin/hostname It knows about paths and options: > grep --ignore-case And history too. Type a command once, and you can re-summon it by just typing a few letters: > rsync -avze ssh . myname@somelonghost.com:/some/long/path/doo/dee/doo/dee/doo To accept the autosuggestion, hit → (right arrow) or Control+F. To accept a single word of the autosuggestion, Alt+→ (right arrow). If the autosuggestion is not what you want, just ignore it. Tab CompletionsA rich set of tab completions work "out of the box".Press Tab and fish will attempt to complete the command, argument, or path: > /priTab => /private/ If there's more than one possibility, it will list them: > ~/stuff/sTab ~/stuff/script.sh (Executable, 4.8kB) ~/stuff/sources/ (Directory) Hit tab again to cycle through the possibilities. fish can also complete many commands, like git branches: > git merge prTab => git merge prompt_designer > git checkout bTab builtin_list_io_merge (Branch) builtin_set_color (Branch) busted_events (Tag) Try hitting tab and see what fish can do! VariablesLike other shells, a dollar sign followed by a variable name is replaced with the value of that variable:> echo My home directory is $HOME My home directory is /home/tutorial This is known as variable substitution, and it also happens in double quotes, but not single quotes: > echo "My current directory is $PWD" My current directory is /home/tutorial > echo 'My current directory is $PWD' My current directory is $PWD Unlike other shells, fish has no dedicated VARIABLE=VALUE syntax for setting variables. Instead it has an ordinary command: set, which takes a variable name, and then its value. > set name 'Mister Noodle' > echo $name Mister Noodle (Notice the quotes: without them, Mister and Noodle would have been separate arguments, and $name would have been made into a list of two elements.) Unlike other shells, variables are not further split after substitution: > mkdir $name > ls Mister Noodle In bash, this would have created two directories "Mister" and "Noodle". In fish, it created only one: the variable had the value "Mister Noodle", so that is the argument that was passed to mkdir, spaces and all. Other shells use the term "arrays", rather than lists. You can erase (or "delete") a variable with -e or --erase > set -e MyVariable > env | grep MyVariable (no output) For more, see Variable expansion. Exports (Shell Variables)Sometimes you need to have a variable available to an external command, often as a setting. For example many programs like git or man read the $PAGER variable to figure out your preferred pager (the program that lets you scroll text). Other variables used like this include $BROWSER, $LANG (to configure your language) and $PATH. You'll note these are written in ALLCAPS, but that's just a convention.To give a variable to an external command, it needs to be "exported". Unlike other shells, fish does not have an export command. Instead, a variable is exported via an option to set, either --export or just -x. > set -x MyVariable SomeValue > env | grep MyVariable MyVariable=SomeValue 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. Exported variables can be local or global or universal - "exported" is not a scope! Usually you'd make them global via set -gx MyVariable SomeValue. For more, see Exporting variables. ListsThe set command above used quotes to ensure that Mister Noodle was one argument. If it had been two arguments, then name would have been a list of length 2. In fact, all variables in fish are really lists, that can contain any number of values, or none at all.Some variables, like $PWD, only have one value. By convention, we talk about that variable's value, but we really mean its first (and only) value. Other variables, like $PATH, really do have multiple values. During variable expansion, the variable expands to become multiple arguments: > echo $PATH /usr/bin /bin /usr/sbin /sbin /usr/local/bin Variables whose name ends in "PATH" are automatically split on colons to become lists. They are joined using colons when exported to subcommands. This is for compatibility with other tools, which expect $PATH to use colons. You can also explicitly add this quirk to a variable with set --path, or remove it with set --unpath. Lists cannot contain other lists: there is no recursion. A variable is a list of strings, full stop. Get the length of a list with count: > count $PATH 5 You can append (or prepend) to a list by setting the list to itself, with some additional arguments. Here we append /usr/local/bin to $PATH: > set PATH $PATH /usr/local/bin You can access individual elements with square brackets. Indexing starts at 1 from the beginning, and -1 from the end: > echo $PATH /usr/bin /bin /usr/sbin /sbin /usr/local/bin > echo $PATH[1] /usr/bin > echo $PATH[-1] /usr/local/bin You can also access ranges of elements, known as "slices": > echo $PATH[1..2] /usr/bin /bin > echo $PATH[-1..2] /usr/local/bin /sbin /usr/sbin /bin You can iterate over a list (or a slice) with a for loop: > for val in $PATH echo "entry: $val" end entry: /usr/bin/ entry: /bin entry: /usr/sbin entry: /sbin entry: /usr/local/bin Lists adjacent to other lists or strings are expanded as cartesian products unless quoted (see Variable expansion): > set a 1 2 3 > set 1 a b c > echo $a$1 1a 2a 3a 1b 2b 3b 1c 2c 3c > echo $a" banana" 1 banana 2 banana 3 banana > echo "$a banana" 1 2 3 banana This is similar to Brace expansion. For more, see Lists. Command SubstitutionsCommand substitutions use the output of one command as an argument to another. Unlike other shells, fish does not use backticks `` for command substitutions. Instead, it uses parentheses:> echo In (pwd), running (uname) In /home/tutorial, running FreeBSD A common idiom is to capture the output of a command in a variable: > set os (uname) > echo $os Linux Command substitutions are not expanded within quotes. Instead, you can temporarily close the quotes, add the command substitution, and reopen them, all in the same argument: > touch "testing_"(date +%s)".txt" > ls *.txt testing_1360099791.txt Unlike other shells, fish does not split command substitutions on any whitespace (like spaces or tabs), only newlines. This can be an issue with commands like pkg-config that print what is meant to be multiple arguments on a single line. To split it on spaces too, use string split. > printf '%s\n' (pkg-config --libs gio-2.0) -lgio-2.0 -lgobject-2.0 -lglib-2.0 > printf '%s\n' (pkg-config --libs gio-2.0 | string split -n " ") -lgio-2.0 -lgobject-2.0 -lglib-2.0 If you need a command substitutions output as one argument, without any splits, use string collect: > echo "first line second line" > myfile > set myfile (cat myfile | string collect) > printf '|%s|' $myfile |first line second line| For more, see Command substitution. Separating Commands (Semicolon)Like other shells, fish allows multiple commands either on separate lines or the same line.To write them on the same line, use the semicolon (";"). That means the following two examples are equivalent: echo fish; echo chips # or echo fish echo chips Exit StatusWhen a command exits, it returns a status code as a non-negative integer.Unlike other shells, fish stores the exit status of the last command in $status instead of $?. > false > echo $status 1 This indicates how the command fared - 0 usually means success, while the others signify kinds of failure. For instance fish's set --query returns the number of variables it queried that weren't set - set --query PATH usually returns 0, set --query arglbargl boogagoogoo usually returns 2. There is also a $pipestatus list variable for the exit statuses [1] of processes in a pipe. For more, see The status variable.
Combiners (And, Or, Not)fish supports the familiar && and || to combine commands, and ! to negate them:> ./configure && make && sudo make install Here, make is only executed if ./configure succeeds (returns 0), and sudo make install is only executed if both ./configure and make succeed. fish also supports and, or, and not. The first two are job modifiers and have lower precedence. Example usage: > cp file1 file1_bak && cp file2 file2_bak; and echo "Backup successful"; or echo "Backup failed" Backup failed As mentioned in the section on the semicolon, this can also be written in multiple lines, like so: cp file1 file1_bak && cp file2 file2_bak and echo "Backup successful" or echo "Backup failed" Conditionals (If, Else, Switch)Use if and else to conditionally execute code, based on the exit status of a command.if grep fish /etc/shells echo Found fish else if grep bash /etc/shells echo Found bash else echo Got nothing end To compare strings or numbers or check file properties (whether a file exists or is writeable and such), use test, like if test "$fish" = "flounder" echo FLOUNDER end # or if test "$number" -gt 5 echo $number is greater than five else echo $number is five or less end # or # This test is true if the path /etc/hosts exists # - it could be a file or directory or symlink (or possibly something else). if test -e /etc/hosts echo We most likely have a hosts file else echo We do not have a hosts file end Combiners can also be used to make more complex conditions, like if grep fish /etc/shells; and command -sq fish echo fish is installed and configured end For even more complex conditions, use begin and end to group parts of them. There is also a switch command: switch (uname) case Linux echo Hi Tux! case Darwin echo Hi Hexley! case FreeBSD NetBSD DragonFly echo Hi Beastie! case '*' echo Hi, stranger! end As you see, case does not fall through, and can accept multiple arguments or (quoted) wildcards. For more, see Conditions. FunctionsA fish function is a list of commands, which may optionally take arguments. Unlike other shells, arguments are not passed in "numbered variables" like $1, but instead in a single list $argv. To create a function, use the function builtin:> function say_hello echo Hello $argv end > say_hello Hello > say_hello everybody! Hello everybody! Unlike other shells, fish does not have aliases or special prompt syntax. Functions take their place. [2] You can list the names of all functions with the functions builtin (note the plural!). fish starts out with a number of functions: > functions N_, abbr, alias, bg, cd, cdh, contains_seq, delete-or-exit, dirh, dirs, disown, down-or-search, edit_command_buffer, export, fg, fish_add_path, fish_breakpoint_prompt, fish_clipboard_copy, fish_clipboard_paste, fish_config, fish_default_key_bindings, fish_default_mode_prompt, fish_git_prompt, fish_hg_prompt, fish_hybrid_key_bindings, fish_indent, fish_is_root_user, fish_job_summary, fish_key_reader, fish_md5, fish_mode_prompt, fish_npm_helper, fish_opt, fish_print_git_action, fish_print_hg_root, fish_prompt, fish_sigtrap_handler, fish_svn_prompt, fish_title, fish_update_completions, fish_vcs_prompt, fish_vi_cursor, fish_vi_key_bindings, funced, funcsave, grep, help, history, hostname, isatty, kill, la, ll, ls, man, nextd, nextd-or-forward-word, open, popd, prevd, prevd-or-backward-word, prompt_hostname, prompt_pwd, psub, pushd, realpath, seq, setenv, suspend, trap, type, umask, up-or-search, vared, wait You can see the source for any function by passing its name to functions: > functions ls function ls --description 'List contents of directory' command ls -G $argv end For more, see Functions.
LoopsWhile loops:> while true echo "Loop forever" end Loop forever Loop forever Loop forever ... # yes, this really will loop forever. Unless you abort it with ctrl-c. For loops can be used to iterate over a list. For example, a list of files: > for file in *.txt cp $file $file.bak end Iterating over a list of numbers can be done with seq: > for x in (seq 5) touch file_$x.txt end For more, see Loops and blocks. PromptUnlike other shells, there is no prompt variable like PS1. To display your prompt, fish executes the fish_prompt function and uses its output as the prompt. And if it exists, fish also executes the fish_right_prompt function and uses its output as the right prompt.You can define your own prompt from the command line: > function fish_prompt; echo "New Prompt % "; end New Prompt % _ Then, if you are happy with it, you can save it to disk by typing funcsave fish_prompt. This saves the prompt in ~/.config/fish/functions/fish_prompt.fish. (Or, if you want, you can create that file manually from the start.) Multiple lines are OK. Colors can be set via set_color, passing it named ANSI colors, or hex RGB values: function fish_prompt set_color purple date "+%m/%d/%y" set_color F00 echo (pwd) '>' (set_color normal) end This prompt would look like: 02/06/13 /home/tutorial > _ You can choose among some sample prompts by running fish_config for a web UI or fish_config prompt for a simpler version inside your terminal. $PATH$PATH is an environment variable containing the directories that fish searches for commands. Unlike other shells, $PATH is a list, 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 Exports. To prepend /usr/local/bin and /usr/sbin to $PATH, you can write: > set PATH /usr/local/bin /usr/sbin $PATH To remove /usr/local/bin from $PATH, you can write: > set PATH (string match -v /usr/local/bin $PATH) For compatibility with other shells and external commands, $PATH is a path variable, and so will be joined with colons (not spaces) when you quote it: > echo "$PATH" /usr/local/sbin:/usr/local/bin:/usr/bin and it will be exported like that, and when fish starts it splits the $PATH it receives into a list on colon. You can do so directly in config.fish, like you might do in other shells with .profile. See this example. A faster way is to use the fish_add_path function, which adds given directories to the path if they aren't already included. It does this by modifying the $fish_user_paths universal variable, which is automatically prepended to $PATH. For example, to permanently add /usr/local/bin to your $PATH, you could write: > fish_add_path /usr/local/bin The advantage is that you don't have to go mucking around in files: just run this once at the command line, and it will affect the current session and all future instances too. You can also add this line to config.fish, as it only adds the component if necessary. Or you can modify $fish_user_paths yourself, but you should be careful not to append to it unconditionally in config.fish, or it will grow longer and longer. Startup (Where's .bashrc?)Fish starts by executing commands in ~/.config/fish/config.fish. You can create it if it does not exist.It is possible to directly create functions and variables in config.fish file, using the commands shown above. For example: > cat ~/.config/fish/config.fish set -x PATH $PATH /sbin/ function ll ls -lh $argv end However, it is more common and efficient to use autoloading functions and universal variables. If you want to organize your configuration, fish also reads commands in .fish files in ~/.config/fish/conf.d/. See Configuration Files for the details. Autoloading FunctionsWhen fish encounters a command, it attempts to autoload a function for that command, by looking for a file with the name of that command in ~/.config/fish/functions/.For example, if you wanted to have a function ll, you would add a text file ll.fish to ~/.config/fish/functions: > cat ~/.config/fish/functions/ll.fish function ll ls -lh $argv end This is the preferred way to define your prompt as well: > cat ~/.config/fish/functions/fish_prompt.fish function fish_prompt echo (pwd) "> " end See the documentation for funced and funcsave for ways to create these files automatically, and $fish_function_path to control their location. Universal VariablesA universal variable is a variable whose value is shared across all instances of fish, now and in the future – even after a reboot. You can make a variable universal with set -U:> set -U EDITOR vim Now in another shell: > echo $EDITOR vim Switching to fish?If you wish to use fish (or any other shell) as your default shell, you need to enter your new shell's executable in two places.Add the shell to /etc/shells with: > echo /usr/local/bin/fish | sudo tee -a /etc/shells Change your default shell with: > chsh -s /usr/local/bin/fish This assumes you installed fish to /usr/local/bin, which is the default location when you've compiled it yourself. If you installed it with a package manager, the usual location is /usr/bin/fish, but package managers typically already add it to /etc/shells. Just substitute the correct location. (To change it back to another shell, just substitute /usr/local/bin/fish with /bin/bash, /bin/tcsh or /bin/zsh as appropriate in the steps above.) Ready for more?If you want to learn more about fish, there is lots of detailed documentation, the official gitter channel, an official mailing list, and the github page.Writing your own completionsTo specify a completion, use the complete command. complete takes as a parameter the name of the command to specify a completion for. For example, to add a completion for the program myprog, one would start the completion command with complete -c myprog ...To provide a list of possible completions for myprog, use the -a switch. If myprog accepts the arguments start and stop, this can be specified as complete -c myprog -a 'start stop'. The argument to the -a switch is always a single string. At completion time, it will be tokenized on spaces and tabs, and variable expansion, command substitution and other forms of parameter expansion will take place. fish has a special syntax to support specifying switches accepted by a command. The switches -s, -l and -o are used to specify a short switch (single character, such as -l), a gnu style long switch (such as --color) and an old-style long switch (like -shuffle), respectively. If the command 'myprog' has an option '-o' which can also be written as --output, and which can take an additional value of either 'yes' or 'no', this can be specified by writing: complete -c myprog -s o -l output -a "yes no" There are also special switches for specifying that a switch requires an argument, to disable filename completion, to create completions that are only available in some combinations, etc.. For a complete description of the various switches accepted by the complete command, see the documentation for the complete builtin, or write complete --help inside the fish shell. As a more comprehensive example, here's a commented excerpt of the completions for systemd's timedatectl: # All subcommands that timedatectl knows - this is useful for later. set -l commands status set-time set-timezone list-timezones set-local-rtc set-ntp # Disable file completions for the entire command # because it does not take files anywhere # Note that this can be undone by using "-F". # # File completions also need to be disabled # if you want to have more control over what files are offered # (e.g. just directories, or just files ending in ".mp3"). complete -c timedatectl -f # This line offers the subcommands # -"status", # -"set-timezone", # -"set-time" # -"list-timezones" # if no subcommand has been given so far. # # The `-n`/`--condition` option takes script as a string, which it executes. # If it returns true, the completion is offered. # Here the condition is the `__fish_seen_subcommands_from` helper function. # If returns true if any of the given commands is used on the commandline, # as determined by a simple heuristic. # For more complex uses, you can write your own function. # See e.g. the git completions for an example. # complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" \ -a "status set-time set-timezone list-timezones" # If the "set-timezone" subcommand is used, # offer the output of `timedatectl list-timezones` as completions. # Each line of output is used as a separate candidate, # and anything after a tab is taken as the description. # It's often useful to transform command output with `string` into that form. complete -c timedatectl -n "__fish_seen_subcommand_from set-timezone" \ -a "(timedatectl list-timezones)" # Completion candidates can also be described via `-d`, # which is useful if the description is constant. # Try to keep these short, because that means the user gets to see more at once. complete -c timedatectl -n "not __fish_seen_subcommand_from $commands" \ -a "set-local-rtc" -d "Maintain RTC in local time" # We can also limit options to certain subcommands by using conditions. complete -c timedatectl -n "__fish_seen_subcommand_from set-local-rtc" \ -l adjust-system-clock -d 'Synchronize system clock from the RTC' # These are simple options that can be used everywhere. complete -c timedatectl -s h -l help -d 'Print a short help text and exit' complete -c timedatectl -l version -d 'Print a short version string and exit' complete -c timedatectl -l no-pager -d 'Do not pipe output into a pager' For examples of how to write your own complex completions, study the completions in /usr/share/fish/completions. (The exact path depends on your chosen installation prefix and may be slightly different) Useful functions for writing completionsfish ships with several functions that are very useful when writing command specific completions. Most of these functions name begins with the string __fish_. Such functions are internal to fish and their name and interface may change in future fish versions. Still, some of them may be very useful when writing completions. A few of these functions are described here. Be aware that they may be removed or changed in future versions of fish.Functions beginning with the string __fish_print_ print a newline separated list of strings. For example, __fish_print_filesystems prints a list of all known file systems. Functions beginning with __fish_complete_ print out a newline separated list of completions with descriptions. The description is separated from the completion by a tab character.
Where to put completionsCompletions can be defined on the commandline or in a configuration file, but they can also be automatically loaded. Fish automatically searches through any directories in the list variable $fish_complete_path, and any completions defined are automatically loaded when needed. A completion file must have a filename consisting of the name of the command to complete and the suffix .fish.By default, Fish searches the following for completions, using the first available file that it finds:
These paths are controlled by parameters set at build, install, or run time, and may vary from the defaults listed above. This wide search may be confusing. If you are unsure, your completions probably belong in ~/.config/fish/completions. If you have written new completions for a common Unix command, please consider sharing your work by submitting it via the instructions in Further help and development If you are developing another program and would like to ship completions with your program, install them to the "vendor" completions directory. As this path may vary from system to system, the pkgconfig framework should be used to discover this path with the output of pkg-config --variable completionsdir fish. DesignThis is a description of the design principles that have been used to design fish. The fish design has three high level goals. These are:
To achieve these high-level goals, the fish design relies on a number of more specific design principles. These are presented below, together with a rationale and a few examples for each. The law of orthogonalityThe shell language should have a small set of orthogonal features. Any situation where two features are related but not identical, one of them should be removed, and the other should be made powerful and general enough to handle all common use cases of either feature.Rationale: Related features make the language larger, which makes it harder to learn. It also increases the size of the source code, making the program harder to maintain and update. Examples:
The law of responsivenessThe shell should attempt to remain responsive to the user at all times, even in the face of contended or unresponsive filesystems. It is only acceptable to block in response to a user initiated action, such as running a command.Rationale: Bad performance increases user-facing complexity, because it trains users to recognize and route around slow use cases. It is also incredibly frustrating. Examples:
Configurability is the root of all evilEvery configuration option in a program is a place where the program is too stupid to figure out for itself what the user really wants, and should be considered a failure of both the program and the programmer who implemented it.Rationale: Different configuration options are a nightmare to maintain, since the number of potential bugs caused by specific configuration combinations quickly becomes an issue. Configuration options often imply assumptions about the code which change when reimplementing the code, causing issues with backwards compatibility. But mostly, configuration options should be avoided since they simply should not exist, as the program should be smart enough to do what is best, or at least a good enough approximation of it. Examples:
A special note on the evils of configurability is the long list of very useful features found in some shells, that are not turned on by default. Both zsh and bash support command-specific completions, but no such completions are shipped with bash by default, and they are turned off by default in zsh. Other features that zsh supports that are disabled by default include tab-completion of strings containing wildcards, a sane completion pager and a history file. The law of user focusWhen designing a program, one should first think about how to make an intuitive and powerful program. Implementation issues should only be considered once a user interface has been designed.Rationale: This design rule is different than the others, since it describes how one should go about designing new features, not what the features should be. The problem with focusing on what can be done, and what is easy to do, is that too much of the implementation is exposed. This means that the user must know a great deal about the underlying system to be able to guess how the shell works, it also means that the language will often be rather low-level. Examples:
The law of discoverabilityA program should be designed to make its features as easy as possible to discover for the user.Rationale: A program whose features are discoverable turns a new user into an expert in a shorter span of time, since the user will become an expert on the program simply by using it. The main benefit of a graphical program over a command-line-based program is discoverability. In a graphical program, one can discover all the common features by simply looking at the user interface and guessing what the different buttons, menus and other widgets do. The traditional way to discover features in command-line programs is through manual pages. This requires both that the user starts to use a different program, and then they remember the new information until the next time they use the same program. Examples:
Release notesfish 3.3.1 (released July 6, 2021)This release of fish fixes the following problems identified in fish 3.3.0:
A number of improvements to the documentation, and fixes for completions, are included as well. If you are upgrading from version 3.2.2 or before, please also review the release notes for 3.3.0 (included below).
fish 3.3.0 (released June 28, 2021)Notable improvements and fixes
Deprecations and removed features
Scripting improvements
Interactive improvements
New or improved bindings
Improved prompts
Completions
Improved terminal support
For distributors
fish 3.2.2 (released April 7, 2021)This release of fish fixes a number of additional issues identified in the fish 3.2 series:
If you are upgrading from version 3.1.2 or before, please also review the release notes for 3.2.1 and 3.2.0 (included below).
fish 3.2.1 (released March 18, 2021)This release of fish fixes the following problems identified in fish 3.2.0:
It also includes some small enhancements:
If you are upgrading from version 3.1.2 or before, please also review the release notes for 3.2.0 (included below).
fish 3.2.0 (released March 1, 2021)Notable improvements and fixes
# Show all dmesg lines related to "usb" dmesg -w | string match '*usb*'
~/dev/build/fish-shell-git/src/fish-shell/build (makepkg)> will turn into: …h-shell/build (makepkg)> It is still possible to react to the COLUMNS variable inside the prompt to implement smarter behavior.
fish_add_path /opt/mycoolthing/bin will add /opt/mycoolthing/bin to the beginning of $fish_user_path without creating duplicates, so it can be called safely from config.fish or interactively, and the path will just be there, once.
> test 1 = 2 and echo true or false test: Expected a combining operator like '-a' at index 4 1 = 2 and echo true or echo false ^ This includes numbering the index from 1 instead of 0, like fish lists.
Syntax changes and new commands
echo $var[1..] echo $var[..-1] echo $var[..] All print the full list $var.
Scripting improvements
Interactive improvements
fish --debug=-warning-path
New or improved bindings
Improved prompts
Improved terminal support
Completions
For distributors
Deprecations and removed features
For distributors and developers
fish 3.1.2 (released April 29, 2020)This release of fish fixes a major issue discovered in fish 3.1.1:
If you are upgrading from version 3.0.0 or before, please also review the release notes for 3.1.1, 3.1.0 and 3.1b1 (included below).
fish 3.1.1 (released April 27, 2020)This release of fish fixes a number of major issues discovered in fish 3.1.0.
This release also includes:
If you are upgrading from version 3.0.0 or before, please also review the release notes for 3.1.0 and 3.1b1 (included below). Errata for fish 3.1A new builtin, time, was introduced in the fish 3.1 releases. This builtin is a reserved word (like test, function, and others) because of the way it is implemented, and functions can no longer be named time. This was not clear in the fish 3.1b1 changelog.
fish 3.1.0 (released February 12, 2020)Compared to the beta release of fish 3.1b1, fish version 3.1.0:
If you are upgrading from version 3.0.0 or before, please also review the release notes for 3.1b1 (included below).
fish 3.1b1 (released January 26, 2020)Notable improvements and fixes
Syntax changes and new commands
Scripting improvements
Interactive improvements
New or improved bindings
Improved prompts
Improved terminal output
Completions
Deprecations and removed features
For distributors and developers
fish 3.0.2 (released February 19, 2019)This release of fish fixes an issue discovered in fish 3.0.1.Fixes and improvements
If you are upgrading from version 2.7.1 or before, please also review the release notes for 3.0.1, 3.0.0 and 3.0b1 (included below). fish 3.0.1 (released February 11, 2019)This release of fish fixes a number of major issues discovered in fish 3.0.0.Fixes and improvements
Known issuesThere is one significant known issue that was not corrected before the release:
If you are upgrading from version 2.7.1 or before, please also review the release notes for 3.0.0 and 3.0b1 (included below).
fish 3.0.0 (released December 28, 2018)fish 3 is a major release, which introduces some breaking changes alongside improved functionality. Although most existing scripts will continue to work, they should be reviewed against the list contained in the 3.0b1 release notes below.Compared to the beta release of fish 3.0b1, fish version 3.0.0:
There is one significant known issue which was not able to be corrected before the release:
If you are upgrading from version 2.7.1 or before, please also review the release notes for 3.0b1 (included below).
fish 3.0b1 (released December 11, 2018)fish 3 is a major release, which introduces some breaking changes alongside improved functionality. Although most existing scripts will continue to work, they should be reviewed against the list below.Notable non-backward compatible changes
DeprecationsWith the release of fish 3, a number of features have been marked for removal in the future. All users are encouraged to explore alternatives. A small number of these features are currently behind feature flags, which are turned on at present but may be turned off by default in the future.A new feature flags mechanism is added for staging deprecations and breaking changes. Feature flags may be specified at launch with fish --features ... or by setting the universal fish_features variable. (#4940)
Notable fixes and improvementsSyntax changes and new commands
New features in commands
Interactive improvements
Other fixes and improvements
For distributors and developers
– fish 2.7.1 (released December 23, 2017)This release of fish fixes an issue where iTerm 2 on macOS would display a warning about paste bracketing being left on when starting a new fish session (#4521).If you are upgrading from version 2.6.0 or before, please also review the release notes for 2.7.0 and 2.7b1 (included below). – fish 2.7.0 (released November 23, 2017)There are no major changes between 2.7b1 and 2.7.0. If you are upgrading from version 2.6.0 or before, please also review the release notes for 2.7b1 (included below).Xcode builds and macOS packages could not be produced with 2.7b1, but this is fixed in 2.7.0. – fish 2.7b1 (released October 31, 2017)Notable improvements
Other significant changes
fish 2.6.0 (released June 3, 2017)Since the beta release of fish 2.6b1, fish version 2.6.0 contains a number of minor fixes, new completions for magneto (#4043), and improvements to the documentation.Known issues
If you are upgrading from version 2.5.0 or before, please also review the release notes for 2.6b1 (included below).
fish 2.6b1 (released May 14, 2017)Notable fixes and improvements
Other significant changes
fish 2.5.0 (released February 3, 2017)There are no major changes between 2.5b1 and 2.5.0. If you are upgrading from version 2.4.0 or before, please also review the release notes for 2.5b1 (included below).Notable fixes and improvements
fish 2.5b1 (released January 14, 2017)Platform ChangesStarting with version 2.5, fish requires a more up-to-date version of C++, specifically C++11 (from 2011). This affects some older platforms:LinuxFor users building from source, GCC’s g++ 4.8 or later, or LLVM’s clang 3.3 or later, are known to work. Older platforms may require a newer compiler installed.Unfortunately, because of the complexity of the toolchain, binary packages are no longer published by the fish-shell developers for the following platforms:
Installing newer version of fish on these systems will require building from source. OS X SnowLeopardStarting with version 2.5, fish requires a C++11 standard library on OS X 10.6 (“SnowLeopard”). If this library is not installed, you will see this error: dyld: Library not loaded: /usr/lib/libc++.1.dylibMacPorts is the easiest way to obtain this library. After installing the SnowLeopard MacPorts release from the install page, run: sudo port -v install libcxx Now fish should launch successfully. (Please open an issue if it does not.) This is only necessary on 10.6. OS X 10.7 and later include the required library by default. Other significant changes
Notable fixes and improvements
fish 2.4.0 (released November 8, 2016)There are no major changes between 2.4b1 and 2.4.0.Notable fixes and improvements
fish 2.4b1 (released October 18, 2016)Significant changes
Notable fixes and improvements
fish 2.3.1 (released July 3, 2016)This is a functionality and bugfix release. This release does not contain all the changes to fish since the last release, but fixes a number of issues directly affecting users at present and includes a small number of new features.Significant changes
Notable fixes and improvements
fish 2.3.0 (released May 20, 2016)There are no significant changes between 2.3.0 and 2.3b2.Other notable fixes and improvements
Known issues
fish 2.3b2 (released May 5, 2016)Significant changes
Other notable fixes and improvements
fish 2.3b1 (released April 19, 2016)Significant Changes
Backward-incompatible changes
Other notable fixes and improvements
fish 2.2.0 (released July 12, 2015)Significant changes
Backward-incompatible changesThese are kept to a minimum, but either change undocumented features or are too hard to use in their existing forms. These changes may break existing scripts.
Other notable fixes and improvements
fish 2.1.2 (released Feb 24, 2015)fish 2.1.2 contains a workaround for a filesystem bug in Mac OS X Yosemite. #1859Specifically, after installing fish 2.1.1 and then rebooting, “Verify Disk” in Disk Utility will report “Invalid number of hard links.” We don’t have any reports of data loss or other adverse consequences. fish 2.1.2 avoids triggering the bug, but does not repair an already affected filesystem. To repair the filesystem, you can boot into Recovery Mode and use Repair Disk from Disk Utility. Linux and versions of OS X prior to Yosemite are believed to be unaffected. There are no other changes in this release.
fish 2.1.1 (released September 26, 2014)Important: if you are upgrading, stop all running instances of fishd as soon as possible after installing this release; it will be restarted automatically. On most systems, there will be no further action required. Note that some environments (where XDG_RUNTIME_DIR is set), such as Fedora 20, will require a restart of all running fish processes before universal variables work as intended.Distributors are highly encouraged to call killall fishd, pkill fishd or similar in installation scripts, or to warn their users to do so. Security fixes
Other fixes
fish 2.1.0Significant Changes
Other Notable Fixes
fish 2.0.0Significant Changes
if set python_path (which python) ... end Because set does not modify $status on success, the if branch effectively tests whether which succeeded, and if so, whether the set also succeeded.
Other Notable Fixes
fishfish Beta r2Bug Fixes
New Features
Credit to @siteshwar for implementation. Thanks @siteshwar!
fishfish Beta r1Scripting
New Features
Programmatic Changes
Performance
Here are some system call counts for launching and then exiting fish with the default configuration, on OS X. The first column is fish trunk, the next column is with our changes, and the last column is bash for comparison. This data was collected via dtrace. before after bash open 9 4 5 fork 28 14 0 stat 131 85 11 lstat 670 0 0 read 332 80 4 write 172 149 0 The large number of forks relative to bash are due to fish’s insanely expensive default prompt, which is unchanged in my version. If we switch to a prompt comparable to bash’s (lame) default, the forks drop to 16 with trunk, 4 after our changes. The large reduction in lstat() numbers is due to fish no longer needing to call ttyname() on OS X. We’ve got some work to do to be as lean as bash, but we’re on the right track. LicenseLicense for fishfish Copyright © 2005-2009 Axel Liljencrantz, 2009-2021 fish-shell contributors. fish is released under the GNU General Public License, version 2.fish includes other code licensed under the GNU General Public License, version 2, including GNU printf. Copyright © 1990-2007 Free Software Foundation, Inc. Printf (from GNU Coreutils 6.9) is released under the GNU General Public License, version 2. The GNU General Public License agreement follows. GNU GENERAL PUBLIC LICENSE Version 2, June 1991 Copyright (C) 1989, 1991 Free Software Foundation, Inc.
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed. Preamble The licenses for most software are designed to take away your freedom to share and change it. By contrast, the GNU General Public License is intended to guarantee your freedom to share and change free software - to make sure the software is free for all its users. This General Public License applies to most of the Free Software Foundation's software and to any other program whose authors commit to using it. (Some other Free Software Foundation software is covered by the GNU Library General Public License instead.) You can apply it to your programs, too. When we speak of free software, we are referring to freedom, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for this service if you wish), that you receive source code or can get it if you want it, that you can change the software or use pieces of it in new free programs; and that you know you can do these things. To protect your rights, we need to make restrictions that forbid anyone to deny you these rights or to ask you to surrender the rights. These restrictions translate to certain responsibilities for you if you distribute copies of the software, or if you modify it. For example, if you distribute copies of such a program, whether gratis or for a fee, you must give the recipients all the rights that you have. You must make sure that they, too, receive or can get the source code. And you must show them these terms so they know their rights. We protect your rights with two steps: (1) copyright the software, and (2) offer you this license which gives you legal permission to copy, distribute and/or modify the software. Also, for each author's protection and ours, we want to make certain that everyone understands that there is no warranty for this free software. If the software is modified by someone else and passed on, we want its recipients to know that what they have is not the original, so that any problems introduced by others will not reflect on the original authors' reputations. Finally, any free program is threatened constantly by software patents. We wish to avoid the danger that redistributors of a free program will individually obtain patent licenses, in effect making the program proprietary. To prevent this, we have made it clear that any patent must be licensed for everyone's free use or not licensed at all. The precise terms and conditions for copying, distribution and modification follow. TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
Activities other than copying, distribution and
modification are not covered by this License; they are outside its scope. The
act of running the Program is not restricted, and the output from the Program
is covered only if its contents constitute a work based on the Program
(independent of having been made by running the Program). Whether that is true
depends on what the Program does.
You may charge a fee for the physical act of transferring
a copy, and you may at your option offer warranty protection in exchange for a
fee.
These requirements apply to the modified work as a whole. If identifiable sections of that work are not derived from the Program, and can be reasonably considered independent and separate works in themselves, then this License, and its terms, do not apply to those sections when you distribute them as separate works. But when you distribute the same sections as part of a whole which is a work based on the Program, the distribution of the whole must be on the terms of this License, whose permissions for other licensees extend to the entire whole, and thus to each and every part regardless of who wrote it. Thus, it is not the intent of this section to claim rights or contest your rights to work written entirely by you; rather, the intent is to exercise the right to control the distribution of derivative or collective works based on the Program. In addition, mere aggregation of another work not based on the Program with the Program (or with a work based on the Program) on a volume of a storage or distribution medium does not bring the other work under the scope of this License.
The source code for a work means the preferred form of the work for making modifications to it. For an executable work, complete source code means all the source code for all modules it contains, plus any associated interface definition files, plus the scripts used to control compilation and installation of the executable. However, as a special exception, the source code distributed need not include anything that is normally distributed (in either source or binary form) with the major components (compiler, kernel, and so on) of the operating system on which the executable runs, unless that component itself accompanies the executable. If distribution of executable or object code is made by offering access to copy from a designated place, then offering equivalent access to copy the source code from the same place counts as distribution of the source code, even though third parties are not compelled to copy the source along with the object code.
If any portion of this section is held invalid or
unenforceable under any particular circumstance, the balance of the section is
intended to apply and the section as a whole is intended to apply in other
circumstances.
It is not the purpose of this section to induce you to infringe any patents or other property right claims or to contest validity of any such claims; this section has the sole purpose of protecting the integrity of the free software distribution system, which is implemented by public license practices. Many people have made generous contributions to the wide range of software distributed through that system in reliance on consistent application of that system; it is up to the author/donor to decide if he or she is willing to distribute software through any other system and a licensee cannot impose that choice. This section is intended to make thoroughly clear what is believed to be a consequence of the rest of this License.
Each version is given a distinguishing version number. If
the Program specifies a version number of this License which applies to it and
"any later version", you have the option of following the terms and
conditions either of that version or of any later version published by the
Free Software Foundation. If the Program does not specify a version number of
this License, you may choose any version ever published by the Free Software
Foundation.
NO WARRANTY
License for PCRE2fish contains code from the [PCRE2](http://www.pcre.org) library to support regular expressions. This code, created by Philip Hazel, is distributed under the terms of the BSD license. Copyright © 1997-2015 University of Cambridge.The BSD license follows. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. License for the Python docs themeIn doc_src/python_docs_theme/, taken from https://pypi.org/project/python-docs-theme/2020.1/.PYTHON SOFTWARE FOUNDATION LICENSE VERSION 2 1. This LICENSE AGREEMENT is between the Python Software Foundation ("PSF"), and the Individual or Organization ("Licensee") accessing and otherwise using this software ("Python") in source or binary form and its associated documentation. 2. Subject to the terms and conditions of this License Agreement, PSF hereby grants Licensee a nonexclusive, royalty-free, world-wide license to reproduce, analyze, test, perform and/or display publicly, prepare derivative works, distribute, and otherwise use Python alone or in any derivative version, provided, however, that PSF's License Agreement and PSF's notice of copyright, i.e., "Copyright (c) 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017 Python Software Foundation; All Rights Reserved" are retained in Python alone or in any derivative version prepared by Licensee. 3. In the event Licensee prepares a derivative work that is based on or incorporates Python or any part thereof, and wants to make the derivative work available to others as provided herein, then Licensee hereby agrees to include in any such work a brief summary of the changes made to Python. 4. PSF is making Python available to Licensee on an "AS IS" basis. PSF MAKES NO REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED. BY WAY OF EXAMPLE, BUT NOT LIMITATION, PSF MAKES NO AND DISCLAIMS ANY REPRESENTATION OR WARRANTY OF MERCHANTABILITY OR FITNESS FOR ANY PARTICULAR PURPOSE OR THAT THE USE OF PYTHON WILL NOT INFRINGE ANY THIRD PARTY RIGHTS. 5. PSF SHALL NOT BE LIABLE TO LICENSEE OR ANY OTHER USERS OF PYTHON FOR ANY INCIDENTAL, SPECIAL, OR CONSEQUENTIAL DAMAGES OR LOSS AS A RESULT OF MODIFYING, DISTRIBUTING, OR OTHERWISE USING PYTHON, OR ANY DERIVATIVE THEREOF, EVEN IF ADVISED OF THE POSSIBILITY THEREOF. 6. This License Agreement will automatically terminate upon a material breach of its terms and conditions. 7. Nothing in this License Agreement shall be deemed to create any relationship of agency, partnership, or joint venture between PSF and Licensee. This License Agreement does not grant permission to use PSF trademarks or trade name in a trademark sense to endorse or promote products or services of Licensee, or any third party. 8. By copying, installing or otherwise using Python, Licensee agrees to be bound by the terms and conditions of this License Agreement. AUTHORfish-shell developersCOPYRIGHT2021, fish-shell developers
Visit the GSP FreeBSD Man Page Interface. |