This wiki is automatically published from ohmyzsh/wiki. To edit this page, go to ohmyzsh/wiki, make your changes and submit a Pull Request.
This page is a description of the current architecture and design of Oh My Zsh.
This is a work in progress: we don't think there's anything outright wrong in it, but much may still be left out.
This page is not authoritative or normative: it was put together by Oh My Zsh users based on the current Oh My Zsh code; it's not a design document from the original Oh My Zsh authors. But it should serve as a useful guide to users and developers who want to get familiar with the Oh My Zsh codebase.
What Oh My Zsh provides:
It seems that plugins can get arbitrarily powerful and do whatever they want, so the user should understand what a given plugin does before enabling it.
These are variables that base OMZ (excluding any plugins) uses. I've read through .oh-my-zsh so far, but not the lib/*.zsh files. More may be on the way.
At initialization time:
In oh-my-zsh.sh:
ZSH - path to .oh-my-zsh (not zsh) installationplugins - user-provided list of plugins to loadZSH_CUSTOM – path to the Oh My Zsh (not zsh itself) customization dirZSH_THEME – theme to load at startupCASE_SENSITIVE – controls zsh completion matchingCOMPLETION_WAITING_DOTSDISABLE_AUTO_UPDATE – ("true"/*)DISABLE_AUTO_PROMPT – ("true"/*)DISABLE_LS_COLORS – in lib/theme-and-appearanceENABLE_CORRECTIONZSH_CACHE_DIRZSH_COMPDUMPZSH_VERSIONZDOTDIRHOMEHOSTOSTYPEThe only required one is $ZSH. The rest either have defaults, or undef is fine and has an expected behavior. (Though if $plugins is unset, the git plugin won't get loaded. Will that break stuff?) All these ZSH_* variables are OMZ-specific, not standard zsh stuff. Except for ZSH_VERSION, which is a standard parameter provided by zsh.
In oh-my-zsh.sh:
At init:
SHORT_HOSTLSCOLORSSCREEN_NOPS1POST_1_7_2_GITPAGERLESSFX – special terminal control "effects" (reset/bold/no-bold/etc)FGBGAt init (defaults if not provided):
ZSH_CUSTOM - defaults to $ZSH/customZSH_CACHE_DIR - defaults to $ZSH/cacheZSH_COMPDUMPZSH_SPECTRUM_TEXTModified at init:
fpathLC_CTYPELeaks:
custom_config_fileURLTOOLS_METHOD - plugins/urltools uses it to manually select node/php/perl/python/etcOh My Zsh is initialized for the current zsh session by sourcing $ZSH/oh-my-zsh.sh. This is typically done from .zshrc, and the Oh My Zsh installation process modifies the user's .zshrc to do so.
The basic steps of the Oh My Zsh initialization process are as follows. Note that the order of steps is subject to change.
The initialization steps in detail:
zsh process$fpath: Add functions/ and completions/$ZSH_CUSTOM and $ZSH_CACHE_DIR$ZSH_CUSTOM/*.zsh file, in alphabetical order$fpath, but don't source)$SHORT_HOST$ZSH_COMPDUMPcompinit, using dump file$plugins, in the order specifiedIn Oh My Zsh terms, customization means adding or overriding zsh code, including its internals and implementation. It's not just a term for user-specified configuration.
Overriding internals can be done by adding *.zsh files to the $ZSH_CUSTOM root directory. All *.zsh files there will be sourced after OMZ loads and sources its own lib/* files. This allows you to redefine functions after the fact. (This will take place after any setup has called OMZ functions.) These are referred to as "config files" in oh-my-zsh.sh.
It's not documented in the Customization page, but $ZSH_CUSTOM/lib/*.zsh do override the corresponding internals lib files. If a custom one is present, it is sourced instead of the one in the distribution.
So, you can:
$plugins after the base plugin$ZSH_CUSTOM controls where the custom override files are found; defaults to $ZSH/custom (under the main OMZ installation).
As of June 2015, user "custom" files are loaded before plugins are loaded, so they cannot be used to modify plugins. To customize a plugin by replacing selected functions or variables, you need to define an additional custom plugin and load that after the base plugin you wish to customize.
The Customization wiki page doesn't discuss which functions/APIs are stable. It mostly talks about overriding on a per-file basis.
Oh My Zsh plugins extend the core functionality of Oh My Zsh.
Plugins provide functionality in the following areas:
A "completion plugin" is the term for a plugin that has nothing but completion system definitions. They are not handled or loaded differently from other plugins.
Plugins are optional, and selected at runtime. When Oh My Zsh is initialized, only the plugins specified in the user-defined $plugins variable are loaded. The core Oh My Zsh code does not depend on any plugins. Themes may depend on plugins. There's no standard mechanism to express these dependencies, though themes should note their plugin dependencies in their comments.
The plugins live in plugins/ in the Oh My Zsh source tree. Even though their source code is in the main Oh My Zsh repository, each plugin has its own maintainer. The maintainers are listed on the [Plugins] page, or in the source code of the plugin.
Themes control the appearance of the zsh prompt, the appearance of certain other programs, and some other behaviors, such as tab and window titles in terminal emulators.
OMZ turns on the prompt_subst shell option, and OMZ themes assume it is enabled.
Themes set a variety of variables to control the appearance of the zsh prompt. They may also install hook functions. These variables are read by core OMZ functions like git_prompt_info() and used to modify their behavior and style their output.
Things themes do:
$PROMPT, $RPROMPT, and related variables$LSCOLORS and $LS_COLORSThese variables are set by themes to control the prompt's appearance and other cosmetic behavior.
PROMPTDEFAULT_USERZSH_THEME_SCM_PROMPT_PREFIX – used in bzr_prompt_info() from lib/bzr.shgit_prompt_info():
ZSH_THEME_GIT_PROMPT_PREFIXZSH_THEME_GIT_PROMPT_SUFFIXZSH_THEME_GIT_COMMITS_AHEAD_PREFIXZSH_THEME_GIT_COMMITS_AHEAD_SUFFIXZSH_THEME_GIT_PROMPT_DIRTYZSH_THEME_GIT_PROMPT_CLEANZSH_THEME_GIT_PROMPT_BEHIND_REMOTEZSH_THEME_GIT_PROMPT_AHEAD_REMOTEZSH_THEME_GIT_PROMPT_DIVERGED_REMOTEZSH_THEME_GIT_PROMPT_AHEADZSH_THEME_GIT_PROMPT_BEHINDZSH_THEME_GIT_PROMPT_SHA_BEFOREZSH_THEME_GIT_PROMPT_SHA_AFTERZSH_THEME_GIT_PROMPT_ADDEDZSH_THEME_GIT_PROMPT_MODIFIEDZSH_THEME_GIT_PROMPT_RENAMEDZSH_THEME_GIT_PROMPT_DELETEDZSH_THEME_GIT_PROMPT_STASHEDZSH_THEME_GIT_PROMPT_UNMERGEDZSH_THEME_GIT_PROMPT_DIVERGEDZSH_THEME_GIT_PROMPT_UNTRACKEDnvm_prompt_info():
ZSH_THEME_NVM_PROMPT_PREFIXZSH_THEME_NVM_PROMPT_SUFFIXrvm_prompt_info():
ZSH_THEME_RVM_PROMPT_PREFIXZSH_THEME_RVM_PROMPT_SUFFIXZSH_THEME_RVM_PROMPT_OPTIONSlib/termsupport.zsh:
ZSH_THEME_TERM_TAB_TITLE_IDLE
ZSH_THEME_TERM_TITLE_IDLE
chpwd_functions
precmd_functions
Not all themes use all of these variables. Most use only a subset.
Other ZSH_THEME_* variables may be used by different themes and plugins. The pattern seems to be that they will supply a XXX_prompt_info() function which can be configured using those variables.
Some per-theme custom variables:
In bureau.zsh-theme:
ZSH_THEME_GIT_PROMPT_STAGEDZSH_THEME_GIT_PROMPT_UNSTAGEDZSH_THEME_GIT_PROMPT_UNTRACKEDThese variables are mostly used in prompt info functions. To use them to customize a theme's prompt, the theme should set the various ZSH_THEME_* variables to contain text that will get interpolated in at the appropriate spots. And then put one or more of these output-capturing function calls inside the PROMPT variable, if you are defining a custom PROMPT:
Or use other *_prompt_info functions that plugins define. These prompt_info functions have dummy implementations (in lib/prompt_info_functions.zsh) so they can be used unconditionally in theme prompts and will gracefully degrade to outputting empty strings if the appropriate plugin is not actually loaded.
The Oh My Zsh prompt construction functions (found inside lib/) send their output to stdout, and are designed to be called with the $(...) output-capturing subshell invocation.
Themes use color definitions from zsh's color definitions. (autoload -U colors && colors).
Some themes may define additional functions, and set precmd or chpwd hooks. There's no defined mechanism for cleaning these up. It looks like themes are assumed to be set up once during initialization, and then not turned off or changed during that session.
Although some existing themes set $chpwd or $precmd, it's probably better for themes to add functions to the $chpwd_functions or $precmd_functions lists, to avoid conflicting with other code that may also want to set hooks.
The Oh My Zsh theme mechanism is designed to load a theme once per session, during OMZ initialization.
The theme mechanism does not provide a way to unload themes. The values for PROMPT, RPROMPT, ZSH_THEME_*, and hooks are not reset. Thus, you can hack in support for switching themes during a session, but it is not clean: when you switch themes, you can get leftover settings from previously loaded themes and end up with a combination of themes.
Plugins or other files that need to locate where they're running from should use ${0:h} instead of a path relative to $ZSH, so they properly find themselves when running from the $ZSH_CUSTOM directory or if their plugin name changes.