360

I'm using zsh terminal, and I'm trying to add a new entry (/home/david/pear/bin) to the PATH variable. I don't see a reference to the PATH variable in my ~/.zshrc file, but doing echo $PATH returns:

/usr/lib/lightdm/lightdm:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games

So I know that the path variable is being set somewhere. Where is the PATH variable set / modified for the zsh terminal?

Jthorpe
  • 8,986
  • 2
  • 44
  • 56
David Barreto
  • 7,687
  • 7
  • 29
  • 45
  • 9
    In my opinion, PATH should be manipulated in `.zshenv`, not in `.zshrc`... – Rmano Sep 30 '20 at 11:01
  • 8
    In case anyone else is curious about @Rmano's pointer on using '.zshenv' (as I was), here's [link](https://unix.stackexchange.com/questions/71253/what-should-shouldnt-go-in-zshenv-zshrc-zlogin-zprofile-zlogout?newreg=741ab675789d4b1ba96862d40c2bb2d7) a detailed discussion. – YCode Jul 04 '21 at 12:53

9 Answers9

440

Actually, using ZSH allows you to use special mapping of environment variables. So you can simply do:

# append
path+=('/home/david/pear/bin')
# or prepend
path=('/home/david/pear/bin' $path)
# export to sub-processes (make it inherited by child processes)
export PATH

For me that's a very neat feature which can be propagated to other variables. Example:

typeset -T LD_LIBRARY_PATH ld_library_path :
Trevor Hickey
  • 34,154
  • 27
  • 144
  • 256
ony
  • 10,885
  • 1
  • 31
  • 41
  • 29
    Nice answer. In my case, `~/.zshrc` is sourced after `.profile`, and overwrites everything in `.profile`. Took a while pulling my hair to figure it out. – Khanh Nguyen Jun 16 '14 at 23:53
  • 2
    The append case does does not need the parens unless you're appending more than one element. It also often doesn't need the quotes. So the simple, short way to append is – Micah Elliott Jun 11 '15 at 21:54
  • 1
    @SuperUberDuper, you should understand that almost any unix shell simply reads startup files which does almost the same as if you'd type it into shell interactively. Regarding "rc" files you might find interesting answer to [this question](http://unix.stackexchange.com/questions/3467/what-does-rc-in-bashrc-stand-for) – ony Mar 07 '16 at 09:08
  • 12
    It's possible to avoid explicit export with `-x` and leave only unique values in a variable with `-U`, colon is assumed by default, so it can be: `typeset -TUx PATH path` – Grief Mar 16 '17 at 19:53
  • 1
    I see the use of `path` and `PATH`, both are separate entities? – CousinCocaine Nov 07 '19 at 15:09
  • @CousinCocaine, you are right. ZSH variables are case-sensitive. `typeset -T` allows to tie `PATH` and `path` together in a special way that internally ZSH manages array variable `path` and reflects its contents in `PATH` as a scalar. – ony Nov 17 '19 at 22:53
  • How do I set exactly what PATH should be, as opposed to appending to it? Because every time `.zshrc` is executed, PATH gets whatever is declared in `.zshrc` appended to it again and again. – Daniel Springer Nov 18 '19 at 00:00
  • @DanielSpringer, there is a hint in second example. By just writing `path=(...)` you get it set. You might be interested as well in comment from @Grief about `-U`. And I'd recommend modifying `PATH` environment from `~/.*profile`-like files so they will not execute when you run inner shell. See also something like https://shreevatsa.wordpress.com/2008/03/30/zshbash-startup-files-loading-order-bashrc-zshrc-etc/ or other similar tips. – ony Nov 24 '19 at 20:04
  • It seems that "path=" appends to path rather then setting it. What I didn't know is that path is reset upon relaunching Terminal/Shell/(maybe?)Login session – Daniel Springer Nov 26 '19 at 02:23
  • 3
    @DanielSpringer, no. If you want it in those terms then: `path=(...)` (without referencing `$path` or `$PATH` inside) assigns, `path=(... $path)` prepends and `path+(...)` appends. – ony Dec 01 '19 at 07:18
  • You can even have it as 1-liner: `export path=(... $path)`. – caram Jan 05 '20 at 10:15
  • This doesn't work for me on my mac. I use the one liner answer below that updates ~/.zshrc insead. – Randy May 05 '20 at 18:02
  • @KhanhNguyen how can I see when these files are sourced? Is there a cmd? – Timo Dec 31 '20 at 10:09
  • 1
    @Timo, in case of zsh you can try something like `zsh -x -i -l -c true` to see all commands during shell start-up. That `-l` is to simulate login-like start (as it happens from text VT or over `ssh`). But you should understand that not always `zsh` will be started as login (e.g. graphical session) and some other parent process may somehow source `.profile` before spawning non-login `zsh` sub-process that inherits environment variables (including `PATH`) from parent process. – ony Jan 01 '21 at 18:37
  • 1
    What if i want to use $HOME inside ' '? path +=('$HOME/etc/etc') doesn't seem to work – Rotkiv Jan 27 '21 at 22:02
  • 1
    @Rotkiv, that's a bit different question about how single/double quotes are interpreted. I guess [zsh expansion](http://zsh.sourceforge.net/Doc/Release/Expansion.html#Parameter-Expansion) covers this. Usually double-quotes allow string interpolation like `"$HOME/etc/etc"`, but single quotes makes it easier to represent text that should not get special treatment. Note that you can always mix in a single argument forms to have something like `"$HOME"'/etc/etc'` (double + single quotes) or `"$HOME"/etc/etc` (double quotes + unquoted). P.S. Yes I'm lazy to try find link to good answer on SO :P – ony Jan 31 '21 at 21:45
  • On macOS, you can prepend and append to your path variable as described in the post, adding to the .zshrc file. It works without exporting the PATH. – Jost Mar 27 '21 at 13:46
  • If this append or prepend syntax doesn't work for you, make sure that you are using lower case `path`. Lower case `path` is the array. – Yerke Jun 17 '21 at 03:35
  • Thank you for showing the prepend as well! Didn't see it anywhere else. – MEMark Aug 21 '21 at 09:38
  • This does not seem to work with `~`. – Minh Nghĩa Jan 14 '22 at 06:54
  • This didn't work for me even though things looked absolutely fine when I `echo $PATH`. I had to go back to the "traditional" syntax of something like: `export PATH=$ANDROID_HOME/platform-tools:$PATH` – Phil Jan 20 '22 at 21:55
  • 1
    @MinhNghĩa, `~` is expanded by shell. If you put it in quotes like `'~/bin'` or `"~/bin"` it will be literally tilda in your path. E.g. you can write `path+=(~/bin)` (note no quotes) or similar. If you need to quote something, you still can do it on part like `path+=(~/'some path with spaces')`. – ony Jan 22 '22 at 17:57
  • @Phil, my best guess for your case is that ok result of `echo $PATH` shows your current shell value for it. Any chance it works if you do `export PATH` (upper case) to export for sub-process after setting `path` (lower case)? – ony Jan 22 '22 at 18:01
355

Here, add this line to .zshrc:

export PATH=/home/david/pear/bin:$PATH

EDIT: This does work, but ony's answer below is better, as it takes advantage of the structured interface ZSH provides for variables like $PATH. This approach is standard for bash, but as far as I know, there is no reason to use it when ZSH provides better alternatives.

Saadat
  • 123
  • 4
  • 14
Linuxios
  • 33,279
  • 13
  • 86
  • 114
  • 35
    haha forget it, I though that was only a console command but adding that line to the .zshrc did the trick. Thanks a lot! – David Barreto Jul 17 '12 at 20:42
  • 1
    My .zshrc already had a line for export PATH so I replaced it with the modified one. – Zack Huston Feb 27 '14 at 13:32
  • 6
    I had to remove the double quotes around the entries i.e. `PATH="/home/david/pear/bin:/usr/bin:etc"` to `PATH=/home/david/pear/bin:/usr/bin:etc` for it to stay in zshrc. – a7omiton Feb 07 '15 at 15:01
  • 4
    @taco, you can use $HOME – mencargo Nov 10 '15 at 01:38
  • It will erase all the old PATH, try `export PATH=$PATH:/path/to/dir` – sstruct Jan 19 '17 at 03:10
  • This appends to my existing PATH, and every time it's run, appends again. How do I set exactly what PATH should be? – Daniel Springer Nov 17 '19 at 23:59
  • You can set `PATH` directly @DanielSpringer: `export PATH=/some/where:/bin:/usr/bin` and so on. Is that what you're asking? The idea is to run the code in this answer in the `.zshrc` or `.profile` or something like that at the beginning of each session. – Linuxios Nov 18 '19 at 00:01
  • 1
    @DanielSpringer: Yes. When you open a shell it inherits `PATH` from the parent process that started it, and then when it runs `.zshrc` (or `.bashrc` or whatever), that's what lets you add extra things to that path. – Linuxios Nov 18 '19 at 00:04
  • @DanielSpringer: Worth noting that you have to add the folder they're *in* to your path, **not** the path to the script file itself. – Linuxios Nov 18 '19 at 00:04
  • @Linuxios Ah, and do I need to add the path to the file itself too, or just its folder? – Daniel Springer Nov 18 '19 at 00:05
  • Just the folder. `PATH` is a list of places (folders; directories) where the shell will look for commands you try to run. For example, when you run `wget ...` the shell is looking for and finding the `wget` binary in `/usr/bin`, for example, which is in the default `PATH`. – Linuxios Nov 18 '19 at 00:07
  • Just a side note, if your path has a space in it. then you need to wrap your path in double quotes like this: ```export PATH=$PATH:"/Applications/Android Studio.app/Contents" ``` – Hamza Waleed Jun 09 '21 at 09:35
93

You can append to your PATH in a minimal fashion. No need for parentheses unless you're appending more than one element. It also usually doesn't need quotes. So the simple, short way to append is:

path+=/some/new/bin/dir

This lower-case syntax is using path as an array, yet also affects its upper-case partner equivalent, PATH (to which it is "bound" via typeset).

(Notice that no : is needed/wanted as a separator.)

Common interactive usage

Then the common pattern for testing a new script/executable becomes:

path+=$PWD/.
# or
path+=$PWD/bin

Common config usage

You can sprinkle path settings around your .zshrc (as above) and it will naturally lead to the earlier listed settings taking precedence (though you may occasionally still want to use the "prepend" form path=(/some/new/bin/dir $path)).

Related tidbits

Treating path this way (as an array) also means: no need to do a rehash to get the newly pathed commands to be found.

Also take a look at vared path as a dynamic way to edit path (and other things).

You may only be interested in path for this question, but since we're talking about exports and arrays, note that arrays generally cannot be exported.

You can even prevent PATH from taking on duplicate entries (refer to this and this):

typeset -U path

PATH pre-populated

The reason your path already has some entries in it is due to your system shell files setting path for you. This is covered in a couple other posts:

Micah Elliott
  • 8,562
  • 4
  • 50
  • 49
  • Should it not be 'path+=:/foo/bar'? (with a colon) – andrew lorien Sep 19 '17 at 07:39
  • 1
    @andrewlorien I updated the answer with details about the colon separator. – Micah Elliott Sep 19 '17 at 17:24
  • Note that if there’s a comment after the path, then we do need quotes, like `path+='my/path' # for fun`. It’s obvious if you have spaces, but not so much if you have comments. – Franklin Yu Mar 06 '19 at 15:05
  • 1
    "(Notice that no : is needed/wanted as a separator.)" This is only true for a lowercase `path`. A preceding `:` is required for `PATH`, as follows in .zshrc `PATH+=:/Users/path/to/my/folder` – Emmett R. Jun 18 '20 at 20:55
  • I use a lot `exec zsh` because I work with plugins from `oh-my-zsh`.Everytime I do exec the path gets bigger. Should I "default" the path from time to time? – Timo Oct 15 '20 at 17:31
  • 1
    For ZSH to interpret $PATH as an array and make it unique, I believe the correct way would be `typeset -aU path` – C. Sederqvist Jul 01 '21 at 22:44
57

one liner, without opening ~/.zshrc file

echo -n 'export PATH=~/bin:$PATH' >> ~/.zshrc

or

echo -n 'export PATH=$HOME/bin:$PATH' >> ~/.zshrc

To see the effect, do source ~/.zshrc in the same tab or open a new tab

Siva Praveen
  • 1,757
  • 15
  • 20
30
  1. Added path to ~/.zshrc

    sudo vi ~/.zshrc

    add new path

    export PATH="$PATH:[NEW_DIRECTORY]/bin"
    
  2. Update ~/.zshrc

    Save ~/.zshrc

    source ~/.zshrc

  3. Check PATH

    echo $PATH

saneryee
  • 2,921
  • 28
  • 20
21

OPTION 1: Add this line to ~/.zshrc:

export "PATH=$HOME/pear/bin:$PATH"

After that you need to run source ~/.zshrc in order your changes to take affect OR close this window and open a new one

OPTION 2: execute it inside the terminal console to add this path only to the current terminal window session. When you close the window/session, it will be lost.

Dimitar
  • 1,703
  • 5
  • 30
  • 47
  • Can you elaborate on how this answer is different from [the same answer posted 5 years ago](https://stackoverflow.com/questions/11530090/adding-a-new-entry-to-the-path-variable-in-zsh/11530176#11530176)? – Franklin Yu Mar 06 '19 at 15:28
  • 1
    in this answer it is not said that you have to add this line of code to the file, if you just run it like that it will change only in the current windows and this is not explain in the original answer – Dimitar Mar 06 '19 at 15:40
1

for me PATH=$PATH:/path/to/file/bin then export PATH worked. to check echo $PATH . other solutions are adding the path temporarily.

Aurora
  • 91
  • 8
0

to verify your new directory has been added correctly, you can use

print -l $path

thanks to the fact that its type is known to be an array

nicolas
  • 9,269
  • 3
  • 35
  • 76
0

If you are on macOS (I'm on Monterey 12.3.1), you may have been pulling your hair like I did metaphorically. These instructions above all worked for me within the terminal session, but I could never get it to persist no matter what I did with export. Moreover, I couldn't find the .zshrc anywhere.

Turns out Apple does it differently. The file you need to edit is etc/paths. You can simply sudo nano /etc/paths and add your path in a new line. Then simply restart terminal and voila.

MNassar
  • 157
  • 2
  • 11