This is my zsh configuration. Well, more a framework of it. It is setting quite a lot of defaults, but at the same time tries to be as user configurable as makes sense. So most of the settings should be easy (enough) to adjust. I’ve taken out the most personal stuff, if you want to see my full config you need to merge this repository together with the one over at git.ganneff.de Git - zsh-ganneff.git/summary
It is hard to say what exact license applies: It consists of a lot of snippets taken from a lot of places all over the net, many of which I don’t remember anymore. Also, many of which are probably small enough that no copyright would apply.
For those I still remember where it is from, I tried to mark them as such, giving credit where its due. Those I know are also listed at the end of this README file. If you find something in here thats from you and want to be mentioned, just mail me.
For stuff that is from me, use the following license terms:
Copyright (C) 2013, Joerg Jaspert <joerg@ganneff.de>
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
To correctly use this fileset for zsh, you need to understand the order in which things are loaded by zsh.
Note: Wherever $ZDOTDIR is unset, $HOME is used instead
- Always
- /etc/zsh/zshenv This is not overridable and not affected by the RCS/GLOBAL_RCS variables explained below!
- $ZDOTDIR/.zshenv
- If the shell is a login shell, it continues with
- /etc/zsh/zprofile
- $ZDOTDIR/.zprofile
- If the shell is interactive, it continues with
- /etc/zsh/zshrc
- $ZDOTDIR/.zshrc
- If the shell is a login shell, it continues with
- /etc/zsh/zlogin
- $ZDOTDIR/zlogin
On logout it is easier: If the shell exits (not due to exec’ing another process, but logout/exit), it reads
- $ZDOTDIR/.zlogout
- /etc/zsh/zlogout
There are 2 variables affecting the above ordering.
- RCS
- If this is unset, none of the (remaining) startup files are sourced.
- GLOBAL_RCS
- If this is unset, none of the startup files in /etc/zsh are sourced.
Note: These variables do NOT affect /etc/zsh/zshenv, which is ALWAYS read!
For the impatient. Assuming you are in your homedirectory, be careful, it will create a .zsh/ and a .zshenv then. You can also put them anywhere else you like, you only have to adjust the .zshenv symlink.
git clone git://git.ganneff.de/zsh.git
ln -sf .zsh/zshenv.home ${HOME}/.zshenv
The config will detect where your checkout is, if you symlinked
${HOME}/.zshenv
to the checked out .zsh/zshenv.home
, so you are not forced
to put it directly in your home (besides the ${HOME}/.zshenv
which is
forced by zsh - somehow it needs to find the rest).
- At startup zsh will read
${HOME}/.zshenv
which sets the environment to load all the rest from${ZDOTDIR}
.${ZDOTDIR}
will be defined as the directory to which the symlink${HOME}/.zshenv
points[fn:1] or${HOME}/.zsh
. All other files are thus loaded from that directory. - Next various variables are defined, ${OSNAME}, ${USER}, ${UID},
${HOST}, ${DOMAIN} and ${DISTRI} and then
${ZDOTDIR}/zshenv.local
is loaded. - ${PATH} as well as ${FPATH} is setup. Only unique entries and
entries that actually exist on the system are kept. That enables us
to define a wide range of possibilities in
${ZDOTDIR}/zshenv.local
- the system will only get what it actually can deal with. - Next file loaded is
${ZDOTDIR}/.zshrc
, which contains the actual magic.- Depending on the zstyles/variables for debug and zrecompile, functions are setup to support them.
- All files in
${ZDOTDIR}
matching the pattern ??_*.zsh are loaded.- The file itself is loaded
- Then we check a number of subdirectories for the “basename” of the file (the name matching the * in the above pattern)[fn:2] and load the files from there. See Variable shell configuration below for details.
- Last we check if the same file, with appended .local exists and load that.[fn:3]
- If it exists,
${ZDOTDIR}/.zshlate
is loaded - If configured too, the ZSH startup time is shown
If the file ${ZDOTDIR}/zshenv.local
exists it will be read at the
very beginning of the zsh startup. At this point only ${ZDOTDIR} and
the basic ${OSNAME}, ${USER}, ${UID}, ${HOST}, ${DOMAIN} and
${DISTRI} have been setup[fn:4], so it is generally not a good idea
to do much in this file.
For that reason the file is kept simple, if (maybe) long. You can
either copy single statements out of ${ZDOTDIR}/zshenv.local.sample
or copy the whole file and then edit it.
In general the values defined in that file are commented there, the following rules apply:
- Commented entries show the default if the option is not given
- Boolean values can be true, yes, on, 1 to enable them, anything else to disable.
- Any other value - see its description in the file
If you got a better name, tell me. But that is basically what we do: Configure zsh based on a series of variables. As already written in Startup described we setup a series of variables and load our configuration based on those. This allows us to overwrite or amend configuration depending on where we are - without having to touch the masterfiles. This configuration framework delivers one actual example for this, if you use it on a Debian system you will find extra aliases dealing with its packaging system.
Using it is simple: Create the right directory, put a file in, restart zsh.
Example: User bob wants to set an extra alias on machines inside the bob-lost.de domain, but only if that machine is running Debian. So he executes:
mkdir -p ${ZDOTDIR}/net:bob-lost.de/distri:Debian
echo 'alias ag=apt-get' >| ${ZDOTDIR}/net:bob-lost.de/distri:Debian/Aliases.zsh
Example 2: User alice wants to adjust the named directory hash table on all her machines, and change one setopt on the machine weirdone.alice-wins.de, but only if that machine is running Debian linux, not if it is booted in kfreebsd or Hurd or RedHat or whatever. So she executes:
mkdir -p ${ZDOTDIR}/net:alice-wins.de/host:weirdone/sys:linux/distri:Debian
echo 'hash -d foo=/home/alice/foo' >| ${ZDOTDIR}/40_Hashes.zsh.local
echo 'setopt beep' >| ${ZDOTDIR}/net:alice-wins.de/host:weirdone/sys:linux/distri:Debian/Options.zsh
Of course those examples are constructed and not entirely real-world
usable. So here is a real one, from me myself and I for you:
On the host franck.debian.org I want a change in my default prompt,
adding one variable information to it. So I have the file
${ZDOTDIR}/net:debian.org/host:franck/Prompts.zsh
with the following
content:
# -*- mode: sh;-*-
# Want one more piece in my prompt here, dinstall status
zstyle ':prompt:ganneff:left:full:setup' items \
ulcorner line openparentheses user at host pts closeparentheses line history \
line dinstall line shell-level line flexline openparentheses path closeparentheses line urcorner newline \
llcorner line rc openparentheses time closeparentheses line vcs line change-root pipe space
zstyle ':prompt:ganneff:extra:dinstall' pre '${PR_CYAN}'
zstyle ':prompt:ganneff:extra:dinstall' post '${PR_NO_COLOR}'
zstyle ':prompt:ganneff:extra:dinstall' token '$DINSTALL'
zstyle ':prompt:ganneff:extra:dinstall' precmd jj_update_dinstall
zmodload zsh/mapfile
jj_update_dinstall () {
DINSTALL="${${(z)${(f)mapfile[/srv/ftp.debian.org/web/dinstall.status]}[2]}[3,-1]}"
}
This framework by default has most of its function included directly, configurable using the zstyle way described already. But there are as many different needs to a shell as there are users of it, so besides all the other flexibility already given, one can easily include “plugins” from wherever.
Plugins are simply files that contain zsh code. Which can do whatever it wants with your shell, so be careful. One exception: Unless the zstyle :ganneff:config:pluginbindkeys is true, plugins are forbidden to bind keys.
Two ways to do it:
- Place the plugin you want to load into
${ZDOTDIR}/plugins
or any subdirectory of it, and add the full path and filename, relative to${ZDOTDIR}/plugins
to the zstyle ganneff:config:plugins in your${ZDOTDIR}/zshenv.local
. - For oh-my-zsh style plugins, put the plugin directory into a
subdirectory of
${ZDOTDIR}/plugins/ohmy
and add the name to the zstyle ganneff:config:ohmyplugins in your${ZDOTDIR}/zshenv.local
The prompt i use is based on various others.
- The “design” is taken from Phil!’s ZSH prompt, as found on http://aperiodic.net/phil/prompt/
- The tech using zstyle is inspired by the prompt as used by the grml-live system http://grml.org, see http://git.grml.org/?p=grml-etc-core.git;a=summary for details on theirs. A good number of their support functions are also taken, though they got renamed from grml_* to ganneff_* to have a single namespace here. (And to direct flames about the bugs to me… :) )
- The winch function as seen in the prompt theme “bart”, delivered with zsh
- The is-{alias,builtin,callable,command,function} and zpgrep function have been taken from “zoppo”, see zoppo for more of it, which is licensed “under WTFPL unless otherwise noted”.
- zbell function is Written by Jean-Philippe Ouellet <jpo@vt.edu> and available under the ISC license, taken from his gist on github
- extract/ls_archive is from Sorin Ionescu and available under the MIT License.
- git-escape-magic is Copyright (c) 2011, 2012 Akinori MUSHA and licensed under the 2-clause BSD license. It is available from his github repository.
[fn:1] Actually, the symlink points to the file zshenv.home inside that directory
[fn:2] For 01_Terminfo.zsh this would be Terminfo.zsh
[fn:3] For 01_Terminfo.zsh this would be, who would have guessed, 01_Terminfo.zsh.local
[fn:4] And maybe ${PS4} and, ${INITLOG} if you turned on tracing in ${HOME}/.zshenv
.