/zsh

Ganneffs zsh config framework

Primary LanguageShell

ZSH configuration “framework”

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

The boring stuff, license / copyright

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.

Usage

Basics

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
    1. /etc/zsh/zshenv This is not overridable and not affected by the RCS/GLOBAL_RCS variables explained below!
    2. $ZDOTDIR/.zshenv
  • If the shell is a login shell, it continues with
    1. /etc/zsh/zprofile
    2. $ZDOTDIR/.zprofile
  • If the shell is interactive, it continues with
    1. /etc/zsh/zshrc
    2. $ZDOTDIR/.zshrc
  • If the shell is a login shell, it continues with
    1. /etc/zsh/zlogin
    2. $ZDOTDIR/zlogin

On logout it is easier: If the shell exits (not due to exec’ing another process, but logout/exit), it reads

  1. $ZDOTDIR/.zlogout
  2. /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!

Howto

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).

Startup described

  1. 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.
  2. Next various variables are defined, ${OSNAME}, ${USER}, ${UID}, ${HOST}, ${DOMAIN} and ${DISTRI} and then ${ZDOTDIR}/zshenv.local is loaded.
  3. ${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.
  4. Next file loaded is ${ZDOTDIR}/.zshrc, which contains the actual magic.
    1. Depending on the zstyles/variables for debug and zrecompile, functions are setup to support them.
    2. 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]
    3. If it exists, ${ZDOTDIR}/.zshlate is loaded
    4. If configured too, the ZSH startup time is shown

zstyle options

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

Variable shell configuration

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]}"
}

Plugins

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:

  1. 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.
  2. 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

Sources

Prompt

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

Various

  • 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.

Footnotes

[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.