How do I get the path of the directory in which a Bash script is located, inside that script?
I want to use a Bash script as a launcher for another application. I want to change the working directory to the one where the Bash script is located, so I can operate on the files in that directory, like so:
$ ./application
82 Answers 82
#!/usr/bin/env bash
SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )
is a useful one-liner which will give you the full directory name of the script no matter where it is being called from.
It will work as long as the last component of the path used to find the script is not a symlink (directory links are OK). If you also want to resolve any links to the script itself, you need a multi-line solution:
#!/usr/bin/env bash
get_script_dir()
{
local SOURCE_PATH="${BASH_SOURCE[0]}"
local SYMLINK_DIR
local SCRIPT_DIR
# Resolve symlinks recursively
while [ -L "$SOURCE_PATH" ]; do
# Get symlink directory
SYMLINK_DIR="$( cd -P "$( dirname "$SOURCE_PATH" )" >/dev/null 2>&1 && pwd )"
# Resolve symlink target (relative or absolute)
SOURCE_PATH="$(readlink "$SOURCE_PATH")"
# Check if candidate path is relative or absolute
if [[ $SOURCE_PATH != /* ]]; then
# Candidate path is relative, resolve to full path
SOURCE_PATH=$SYMLINK_DIR/$SOURCE_PATH
fi
done
# Get final script directory path from fully resolved source path
SCRIPT_DIR="$(cd -P "$( dirname "$SOURCE_PATH" )" >/dev/null 2>&1 && pwd)"
echo "$SCRIPT_DIR"
}
echo "get_script_dir: $(get_script_dir)"
This last one will work with any combination of aliases, source
, bash -c
, symlinks, etc.
Beware: if you cd
to a different directory before running this snippet, the result may be incorrect!
Also, watch out for $CDPATH
gotchas, and stderr output side effects if the user has smartly overridden cd to redirect output to stderr instead (including escape sequences, such as when calling update_terminal_cwd >&2
on Mac). Adding >/dev/null 2>&1
at the end of your cd
command will take care of both possibilities.
To understand how it works, try running this more verbose form:
#!/usr/bin/env bash
SOURCE=${BASH_SOURCE[0]}
while [ -L "$SOURCE" ]; do # resolve $SOURCE until the file is no longer a symlink
TARGET=$(readlink "$SOURCE")
if [[ $TARGET == /* ]]; then
echo "SOURCE '$SOURCE' is an absolute symlink to '$TARGET'"
SOURCE=$TARGET
else
DIR=$( dirname "$SOURCE" )
echo "SOURCE '$SOURCE' is a relative symlink to '$TARGET' (relative to '$DIR')"
SOURCE=$DIR/$TARGET # if $SOURCE was a relative symlink, we need to resolve it relative to the path where the symlink file was located
fi
done
echo "SOURCE is '$SOURCE'"
RDIR=$( dirname "$SOURCE" )
DIR=$( cd -P "$( dirname "$SOURCE" )" >/dev/null 2>&1 && pwd )
if [ "$DIR" != "$RDIR" ]; then
echo "DIR '$RDIR' resolves to '$DIR'"
fi
echo "DIR is '$DIR'"
And it will print something like:
SOURCE './scriptdir.sh' is a relative symlink to 'sym2/scriptdir.sh' (relative to '.')
SOURCE is './sym2/scriptdir.sh'
DIR './sym2' resolves to '/home/ubuntu/dotfiles/fo fo/real/real1/real2'
DIR is '/home/ubuntu/dotfiles/fo fo/real/real1/real2'
18 Comments
source <script>
and bash <script>
: DIR="$(cd -P "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
.cd
prints something to STDOUT! E.g., if your $CDPATH
has .
. To cover this case, use DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" > /dev/null && pwd )"
dirname $(readlink -f 0ドル)
is the right command. See gist.github.com/tvlooy/cbfbdb111a4ebad8b93e for a testcasedirname "$(readlink -f "0ドル")"
doesn't add complexity and is fair measure more robust for the minimal amount of trouble.dirname "$(readlink -f "0ドル")"
fails though in the case where the script is sourced, eg: /bin/bash -c . script.sh
... use $(dirname "$(readlink -f "${BASH_SOURCE[0]}")")
insteadUse dirname "0ドル"
:
test.sh:
#!/usr/bin/env bash
echo "The script you are running has:"
echo "basename: [$(basename "0ドル")]"
echo "dirname : [$(dirname "0ドル")]"
echo "pwd : [$(pwd)]"
Using pwd
alone will not work if you are not running the script from the directory it is contained in.
[~]$ pwd
/home/matt
[~]$ ./test.sh
The script you are running has:
basename: [test.sh]
dirname : [/home/matt]
pwd : [/home/matt]
[~]$ cd /tmp
[~/tmp]$ ~/test.sh
The script you are running has:
basename: [test.sh]
dirname : [/home/matt]
pwd : [/tmp]
11 Comments
type -p
if the script is executable. This can also open a subtle hole if the script is executed using bash test2.sh
and there is another script with the same name executable somewhere else.bash
and the hash-bang line explicitly mentions /bin/bash
I'd say it's pretty safe to depend on bashisms.dirname 0ドル
is that if the directory is the current directory, you'll get .
. That's fine unless you're going to change directories in the script and expect to use the path you got from dirname 0ドル
as though it were absolute. To get the absolute path: pushd `dirname 0ドル` > /dev/null
, SCRIPTPATH=`pwd`
, popd > /dev/null
: pastie.org/1489386 (But surely there's a better way to expand that path?)dirname 0ドル
is a problem if you assign it to a variable and then use it to launch a script like $dir/script.sh
; I would imagine this is the use case for this type of thing 90% of the time. ./script.sh
would work fine.The dirname
command is the most basic, simply parsing the path up to the filename off of the 0ドル
(script name) variable:
dirname -- "0ドル";
But, as matt b pointed out, the path returned is different depending on how the script is called. pwd
doesn't do the job because that only tells you what the current directory is, not what directory the script resides in. Additionally, if a symbolic link to a script is executed, you're going to get a (probably relative) path to where the link resides, not the actual script.
Some others have mentioned the readlink
command, but at its simplest, you can use:
dirname -- "$( readlink -f -- "0ドル"; )";
readlink
will resolve the script path to an absolute path from the root of the filesystem. So, any paths containing single or double dots, tildes and/or symbolic links will be resolved to a full path.
Here's a script demonstrating each of these, whatdir.sh
:
#!/usr/bin/env bash
echo "pwd: `pwd`"
echo "\0ドル: 0ドル"
echo "basename: `basename -- "0ドル"`"
echo "dirname: `dirname -- "0ドル"`"
echo "dirname/readlink: $( dirname -- "$( readlink -f -- "0ドル"; )"; )"
Running this script in my home dir, using a relative path:
>>>$ ./whatdir.sh
pwd: /Users/phatblat
0ドル: ./whatdir.sh
basename: whatdir.sh
dirname: .
dirname/readlink: /Users/phatblat
Again, but using the full path to the script:
>>>$ /Users/phatblat/whatdir.sh
pwd: /Users/phatblat
0ドル: /Users/phatblat/whatdir.sh
basename: whatdir.sh
dirname: /Users/phatblat
dirname/readlink: /Users/phatblat
Now changing directories:
>>>$ cd /tmp
>>>$ ~/whatdir.sh
pwd: /tmp
0ドル: /Users/phatblat/whatdir.sh
basename: whatdir.sh
dirname: /Users/phatblat
dirname/readlink: /Users/phatblat
And finally using a symbolic link to execute the script:
>>>$ ln -s ~/whatdir.sh whatdirlink.sh
>>>$ ./whatdirlink.sh
pwd: /tmp
0ドル: ./whatdirlink.sh
basename: whatdirlink.sh
dirname: .
dirname/readlink: /Users/phatblat
There is however one case where this doesn't work, when the script is sourced (instead of executed) in bash:
>>>$ cd /tmp
>>>$ . ~/whatdir.sh
pwd: /tmp
0ドル: bash
basename: bash
dirname: .
dirname/readlink: /tmp
15 Comments
readlink
will not availabe in some platform in default installation. Try to avoid using it if you canexport SCRIPT_DIR="$(dirname "$(readlink -f "0ドル")")"
-f
is not recognised as an option to readlink
. Using stat -f
instead does the job. Thanksgreadlink
, which is basically the readlink
we are all familiar. Here is a platform independent version: dir=`greadlink -f ${BASH_SOURCE[0]} || readlink -f ${BASH_SOURCE[0]}`
0ドル
doesn't work if the file is sourced. You get -bash
instead of the script name.pushd . > '/dev/null';
SCRIPT_PATH="${BASH_SOURCE[0]:-0ドル}";
while [ -h "$SCRIPT_PATH" ];
do
cd "$( dirname -- "$SCRIPT_PATH"; )";
SCRIPT_PATH="$( readlink -f -- "$SCRIPT_PATH"; )";
done
cd "$( dirname -- "$SCRIPT_PATH"; )" > '/dev/null';
SCRIPT_PATH="$( pwd; )";
popd > '/dev/null';
It works for all versions, including
- when called via multiple depth soft link,
- when the file it
- when script called by command "
source
" aka.
(dot) operator. - when arg
0ドル
is modified from caller. "./script"
"/full/path/to/script"
"/some/path/../../another/path/script"
"./some/folder/script"
Alternatively, if the Bash script itself is a relative symlink you want to follow it and return the full path of the linked-to script:
pushd . > '/dev/null';
SCRIPT_PATH="${BASH_SOURCE[0]:-0ドル}";
while [ -h "$SCRIPT_PATH" ];
do
cd "$( dirname -- "$SCRIPT_PATH"; )";
SCRIPT_PATH="$( readlink -f -- "$SCRIPT_PATH"; )";
done
cd "$( dirname -- "$SCRIPT_PATH"; )" > '/dev/null';
SCRIPT_PATH="$( pwd; )";
popd > '/dev/null';
SCRIPT_PATH
is given in full path, no matter how it is called.
Just make sure you locate this at start of the script.
7 Comments
readlink -f $(dirname "${VIRTUAL_ENV}")
;dirname "${SCRIPT_PATH}"
&& pwd)? But anyway great script!cd
out of its current directory in the hope of cd
ing back again later: The script may not have permission to change directory back to the directory that was current when it was invoked. (Same goes for pushd/popd)readlink -f
is GNU-specific. BSD readlink
does not have that option.Here is an easy-to-remember script:
DIR="$( dirname -- "${BASH_SOURCE[0]}"; )"; # Get the directory name
DIR="$( realpath -e -- "$DIR"; )"; # Resolve its full path if need be
9 Comments
DIR=$(realpath "$(dirname "${BASH_SOURCE[0]}")")
realpath
from resolving "manually" with a loop of readlink
? Even the readlink
man page says Note realpath(1) is the preferred command to use for canonicalization functionality.
realpath
before dirname
, not after? If the script file itself is a symlink... It would give something like DIR="$(dirname "$(realpath "${BASH_SOURCE[0]}")")"
. Actually very close to the answer proposed by Simon.realpath
(and man7.org/linux/man-pages/man1/realpath.1.html ) don't have the -f
option; is that supposed to be -e
@vaeVictis ? (That user added it.)You can use $BASH_SOURCE
:
#!/usr/bin/env bash
scriptdir="$( dirname -- "$BASH_SOURCE"; )";
Note that you need to use #!/bin/bash
and not #!/bin/sh
since it's a Bash extension.
5 Comments
./foo/script
, then $(dirname $BASH_SOURCE)
is ./foo
.realpath
command to get full path of ./foo/script. So dirname $(realpath ./foo/script)
will give the path of script.BASH_SOURCE
is an array, so I'm not sure this is right. All the other answers use BASH_SOURCE[0]
which seems more plausible.source
scriptdir=`dirname "${BASH_SOURCE[0]}"`
This should do it:
DIR="$(dirname "$(realpath "0ドル")")"
This works with symlinks and spaces in path.
9 Comments
./script.sh
shows .
instead of the full directory pathstat
instead. But still, it shows .
if you are in 'this' dir.coreutils
from Homebrew and use greadlink
to get the -f
option on MacOS because it is *BSD under the covers and not Linux.realpath: command not found
on vanilla macOS, I suppose you have installed some homebrew package to get it working on your mac$BASH_SOURCE
works properly in my .bashrc
while 0ドル
does not7 Comments
source
on these and cd $(dirname 0ドル)
is easy to remember.${BASH_SOURCE[0]}
instead of 0ドル
will work with source my/script.sh
${BASH_SOURCE[0]}
is not satisfactory at all. ${BASH_SOURCE:-0}
is much better.pwd
can be used to find the current working directory, and dirname
to find the directory of a particular file (command that was run, is 0ドル
, so dirname 0ドル
should give you the directory of the current script).
However, dirname
gives precisely the directory portion of the filename, which more likely than not is going to be relative to the current working directory. If your script needs to change directory for some reason, then the output from dirname
becomes meaningless.
I suggest the following:
#!/usr/bin/env bash
reldir="$( dirname -- "0ドル"; )";
cd "$reldir";
directory="$( pwd; )";
echo "Directory is ${directory}";
This way, you get an absolute, rather than a relative directory.
Since the script will be run in a separate Bash instance, there isn't any need to restore the working directory afterwards, but if you do want to change back in your script for some reason, you can easily assign the value of pwd
to a variable before you change directory, for future use.
Although just
cd "$( dirname -- "0ドル"; )";
solves the specific scenario in the question, I find having the absolute path to more more useful generally.
2 Comments
dirname 0ドル
&& pwd)This gets the current working directory on Mac OS X v10.6.6 (Snow Leopard):
DIR=$(cd "$(dirname "0ドル")"; pwd)
1 Comment
SCRIPT_DIR=$( cd ${0%/*} && pwd -P )
2 Comments
0ドル
nor pwd
are guaranteed to have the right information, depending on how the script is invoked.I don't think this is as easy as others have made it out to be. pwd
doesn't work, as the current directory is not necessarily the directory with the script. 0ドル
doesn't always have the information either. Consider the following three ways to invoke a script:
./script
/usr/bin/script
script
In the first and third ways 0ドル
doesn't have the full path information. In the second and third, pwd
does not work. The only way to get the directory in the third way would be to run through the path and find the file with the correct match. Basically the code would have to redo what the OS does.
One way to do what you are asking would be to just hardcode the data in the /usr/share
directory, and reference it by its full path. Data shoudn't be in the /usr/bin
directory anyway, so this is probably the thing to do.
3 Comments
mydir=$(realpath 0ドル)
. I've tested that it works in all the situations you describe.$(dirname "$(readlink -f "$BASH_SOURCE")")
2 Comments
$BASH_SOURCE
over 0ドル
, because it's explicit even for readers not well-versed in bash. $(dirname -- "$(readlink -f -- "$BASH_SOURCE")")
$BASH_SOURCE
works while 0ドル
does not in case of my .bashrc
(where both symlink AND sourcing is also used)For Python, see my other answer here.
For Bash, see below:
Summary:
Here is a full program:
my_script.sh:
#!/usr/bin/env bash
# (Commented out because using index 0, as shown farther below, is generally
# better)
# FULL_PATH_TO_SCRIPT="$(realpath "${BASH_SOURCE[-1]}")"
# OR, if you do NOT need it to work for **sourced** scripts too:
# FULL_PATH_TO_SCRIPT="$(realpath "0ドル")"
# [Generally prefer this, using index `0`, over the one above using
# index `-1`]
#
# OR, depending on which path you want, in case of nested `source` calls,
# use index `0` instead of `-1`:
# - NB: this may fix the error `No such file or directory` when sourcing
# this script from another script, where this script sources another
# file based on the path to its own `FULL_PATH_TO_SCRIPT`, but so does
# the other script sourcing this script, based on its own outer
# `FULL_PATH_TO_SCRIPT`.
FULL_PATH_TO_SCRIPT="$(realpath "${BASH_SOURCE[0]}")"
# OR, add `-s` to NOT expand symlinks in the path:
# FULL_PATH_TO_SCRIPT="$(realpath -s "${BASH_SOURCE[-1]}")"
SCRIPT_DIRECTORY="$(dirname "$FULL_PATH_TO_SCRIPT")"
SCRIPT_FILENAME="$(basename "$FULL_PATH_TO_SCRIPT")"
# Going further: also obtain the filename with and without the extension
SCRIPT_FILENAME_STEM="${SCRIPT_FILENAME%.*}" # withOUT the extension
SCRIPT_FILENAME_EXTENSION="${SCRIPT_FILENAME##*.}" # JUST the extension
# debug prints of all of the above
echo "FULL_PATH_TO_SCRIPT: $FULL_PATH_TO_SCRIPT"
echo "SCRIPT_DIRECTORY: $SCRIPT_DIRECTORY"
echo "SCRIPT_FILENAME: $SCRIPT_FILENAME"
echo "SCRIPT_FILENAME_STEM: $SCRIPT_FILENAME_STEM"
echo "SCRIPT_FILENAME_EXTENSION: $SCRIPT_FILENAME_EXTENSION"
Mark your program above as executable:
chmod +x my_script.sh
Run it:
./my_script.sh
Example run command and output:
~/GS/dev/eRCaGuy_PathShortener$ ./my_script.sh
FULL_PATH_TO_SCRIPT: /home/gabriel/GS/dev/eRCaGuy_PathShortener/my_script.sh
SCRIPT_DIRECTORY: /home/gabriel/GS/dev/eRCaGuy_PathShortener
SCRIPT_FILENAME: my_script.sh
SCRIPT_FILENAME_STEM: my_script
SCRIPT_FILENAME_EXTENSION: sh
Details:
How to obtain the full file path, full directory, and base filename of any script being run OR sourced...
...even when the called script is called from within another bash function or script, or when nested sourcing is being used!
For many cases, all you need to acquire is the full path to the script you just called. This can be easily accomplished using realpath
. Note that realpath
is part of GNU coreutils. If you don't have it already installed (it comes default on Ubuntu), you can install it with sudo apt update && sudo apt install coreutils
.
get_script_path.sh (for the latest version of this script, see get_script_path.sh in my eRCaGuy_hello_world repo):
#!/bin/bash
# A. Obtain the full path, and expand (walk down) symbolic links
# A.1. `"0ドル"` works only if the file is **run**, but NOT if it is **sourced**.
# FULL_PATH_TO_SCRIPT="$(realpath "0ドル")"
# A.2. `"${BASH_SOURCE[-1]}"` works whether the file is sourced OR run, and even
# if the script is called from within another bash function!
# NB: if `"${BASH_SOURCE[-1]}"` doesn't give you quite what you want, use
# `"${BASH_SOURCE[0]}"` instead in order to get the first element from the array.
FULL_PATH_TO_SCRIPT="$(realpath "${BASH_SOURCE[-1]}")"
# B.1. `"0ドル"` works only if the file is **run**, but NOT if it is **sourced**.
# FULL_PATH_TO_SCRIPT_KEEP_SYMLINKS="$(realpath -s "0ドル")"
# B.2. `"${BASH_SOURCE[-1]}"` works whether the file is sourced OR run, and even
# if the script is called from within another bash function!
# NB: if `"${BASH_SOURCE[-1]}"` doesn't give you quite what you want, use
# `"${BASH_SOURCE[0]}"` instead in order to get the first element from the array.
FULL_PATH_TO_SCRIPT_KEEP_SYMLINKS="$(realpath -s "${BASH_SOURCE[-1]}")"
# You can then also get the full path to the directory, and the base
# filename, like this:
SCRIPT_DIRECTORY="$(dirname "$FULL_PATH_TO_SCRIPT")"
SCRIPT_FILENAME="$(basename "$FULL_PATH_TO_SCRIPT")"
# Now print it all out
echo "FULL_PATH_TO_SCRIPT = \"$FULL_PATH_TO_SCRIPT\""
echo "SCRIPT_DIRECTORY = \"$SCRIPT_DIRECTORY\""
echo "SCRIPT_FILENAME = \"$SCRIPT_FILENAME\""
IMPORTANT note on nested source
calls: if "${BASH_SOURCE[-1]}"
above doesn't give you quite what you want, try using "${BASH_SOURCE[0]}"
instead. The first (0
) index gives you the first entry in the array, and the last (-1
) index gives you the last last entry in the array. Depending on what it is you're after, you may actually want the first entry. I discovered this to be the case when I sourced ~/.bashrc
with . ~/.bashrc
, which sourced ~/.bash_aliases
with . ~/.bash_aliases
, and I wanted the realpath
(with expanded symlinks) to the ~/.bash_aliases
file, NOT to the ~/.bashrc
file. Since these are nested source
calls, using "${BASH_SOURCE[0]}"
gave me what I wanted: the expanded path to ~/.bash_aliases
! Using "${BASH_SOURCE[-1]}"
, however, gave me what I did not want: the expanded path to ~/.bashrc
.
Example command and output:
- Running the script:
~/GS/dev/eRCaGuy_hello_world/bash$ ./get_script_path.sh FULL_PATH_TO_SCRIPT = "/home/gabriel/GS/dev/eRCaGuy_hello_world/bash/get_script_path.sh" SCRIPT_DIRECTORY = "/home/gabriel/GS/dev/eRCaGuy_hello_world/bash" SCRIPT_FILENAME = "get_script_path.sh"
- Sourcing the script with
. get_script_path.sh
orsource get_script_path.sh
(the result is the exact same as above because I used"${BASH_SOURCE[-1]}"
in the script instead of"0ドル"
):~/GS/dev/eRCaGuy_hello_world/bash$ . get_script_path.sh FULL_PATH_TO_SCRIPT = "/home/gabriel/GS/dev/eRCaGuy_hello_world/bash/get_script_path.sh" SCRIPT_DIRECTORY = "/home/gabriel/GS/dev/eRCaGuy_hello_world/bash" SCRIPT_FILENAME = "get_script_path.sh"
If you use "0ドル"
in the script instead of "${BASH_SOURCE[-1]}"
, you'll get the same output as above when running the script, but this undesired output instead when sourcing the script:
~/GS/dev/eRCaGuy_hello_world/bash$ . get_script_path.sh
FULL_PATH_TO_SCRIPT = "/bin/bash"
SCRIPT_DIRECTORY = "/bin"
SCRIPT_FILENAME = "bash"
And, apparently if you use "$BASH_SOURCE"
instead of "${BASH_SOURCE[-1]}"
, it will not work if the script is called from within another bash function. So, using "${BASH_SOURCE[-1]}"
is therefore the best way to do it, as it solves both of these problems! See the references below.
Difference between realpath
and realpath -s
:
Note that realpath
also successfully walks down symbolic links to determine and point to their targets rather than pointing to the symbolic link. If you do NOT want this behavior (sometimes I don't), then add -s
to the realpath
command above, making that line look like this instead:
# Obtain the full path, but do NOT expand (walk down) symbolic links; in
# other words: **keep** the symlinks as part of the path!
FULL_PATH_TO_SCRIPT="$(realpath -s "${BASH_SOURCE[-1]}")"
This way, symbolic links are NOT expanded. Rather, they are left as-is, as symbolic links in the full path.
The code above is now part of my eRCaGuy_hello_world repo in this file here: bash/get_script_path.sh. Reference and run this file for full examples both with and withOUT symlinks in the paths. See the bottom of the file for example output in both cases.
References:
- How to retrieve absolute path given relative
- taught me about the
BASH_SOURCE
variable: Unix & Linux: determining path to sourced shell script - taught me that
BASH_SOURCE
is actually an array, and we want the last element from it for it to work as expected inside a function (hence why I used"${BASH_SOURCE[-1]}"
in my code here): Unix & Linux: determining path to sourced shell script man bash
--> search forBASH_SOURCE
:BASH_SOURCE
An array variable whose members are the source filenames where the corresponding shell function names in the
FUNCNAME
array variable are defined. The shell function${FUNCNAME[$i]}
is defined in the file${BASH_SOURCE[$i]}
and called from${BASH_SOURCE[$i+1]}
.
See also:
- My answer for Python: How do I get the path and name of the python file that is currently executing?
- [my answer] Unix & Linux: determining path to sourced shell script
3 Comments
${BASH_SOURCE[-1]}
and ${BASH_SOURCE[0]}
? I know -1
retrieves the last element from the array and 0
retrieves the first one but in which case do I want to use one over the other ?IMPORTANT note on nested source calls
section in the answer. It has to do with nested sourcing, when one script which you source sources another script.1
would have given me the same result as -1
since I believe the array only had 2 elements in it, so that would have been the last element in both cases.This is Linux specific, but you could use:
SELF=$(readlink /proc/$$/fd/255)
2 Comments
/proc/fd/$$/255
seems to point to the tty, not to a directory. For example, in my current login shell, file descriptors 0, 1, 2, and 255 all refer to /dev/pts/4
. In any case, the bash manual doesn't mention fd 255, so it's probably unwise to depend on this behavior.\realpath ${BASH_SOURCE[0]};
would seem to be the best way to go.Here is a POSIX compliant one-liner:
SCRIPT_PATH=`dirname "0ドル"`; SCRIPT_PATH=`eval "cd \"$SCRIPT_PATH\" && pwd"`
# test
echo $SCRIPT_PATH
2 Comments
cd
is configured to print the new path name.The shortest and most elegant way to do this is:
#!/bin/bash
DIRECTORY=$(cd `dirname 0ドル` && pwd)
echo $DIRECTORY
This would work on all platforms and is super clean.
More details can be found in "Which directory is that bash script in?".
1 Comment
#!/bin/sh
PRG="0ドル"
# need this for relative symlinks
while [ -h "$PRG" ] ; do
PRG=`readlink "$PRG"`
done
scriptdir=`dirname "$PRG"`
1 Comment
Try using:
real=$(realpath "$(dirname "0ドル")")
7 Comments
0ドル
for ${BASH_SOURCE[0]}
so that this method will work anywhere, including in a function.dirname
because the last part of 0ドル
may be a symlink that points to a file that is not in the same directory as the symlink itself. The solution described in this answer just gets the path of the directory where the symlink it stored, not the directory of the target. Furthermore, this solution is missing quoting. It will not work if the path contains special characters.dir="$(realpath "$(dirname "${BASH_SOURCE[0]}")")"
Here is the simple, correct way:
actual_path=$(readlink -f "${BASH_SOURCE[0]}")
script_dir=$(dirname "$actual_path")
Explanation:
${BASH_SOURCE[0]}
- the full path to the script. The value of this will be correct even when the script is being sourced, e.g.source <(echo 'echo 0ドル')
prints bash, while replacing it with${BASH_SOURCE[0]}
will print the full path of the script. (Of course, this assumes you're OK taking a dependency on Bash.)readlink -f
- Recursively resolves any symlinks in the specified path. This is a GNU extension, and not available on (for example) BSD systems. If you're running a Mac, you can use Homebrew to install GNUcoreutils
and supplant this withgreadlink -f
.And of course
dirname
gets the parent directory of the path.
1 Comment
greadlink -f
unfortunately doesn't work effectively when source
ing the script on Mac :(I tried all of these and none worked. One was very close, but it had a tiny bug that broke it badly; they forgot to wrap the path in quotation marks.
Also a lot of people assume you're running the script from a shell, so they forget when you open a new script it defaults to your home.
Try this directory on for size:
/var/No one/Thought/About Spaces Being/In a Directory/Name/And Here's your file.text
This gets it right regardless how or where you run it:
#!/bin/bash
echo "pwd: `pwd`"
echo "\0ドル: 0ドル"
echo "basename: `basename "0ドル"`"
echo "dirname: `dirname "0ドル"`"
So to make it actually useful, here's how to change to the directory of the running script:
cd "`dirname "0ドル"`"
2 Comments
ln -s ../bin64/foo /usr/bin/foo
).This is a slight revision to the solution e-satis and 3bcdnlklvc04a pointed out in their answer:
SCRIPT_DIR=''
pushd "$(dirname "$(readlink -f "$BASH_SOURCE")")" > /dev/null && {
SCRIPT_DIR="$PWD"
popd > /dev/null
}
This should still work in all the cases they listed.
This will prevent popd
after a failed pushd
. Thanks to konsolebox.
5 Comments
SCRIPT_DIR=''; pushd "$(dirname "$(readlink -f "$BASH_SOURCE")")" > /dev/null && { SCRIPT_DIR=$PWD; popd > /dev/null; }
popd
in cases (even when rare) where pushd
fails. And in case pushd
fails, what do you think should be the value of SCRIPT_DIR
? The action may vary depending on what may seem logical or what one user could prefer but certainly, doing popd
is wrong.pushd
popd
dangers could be avoided simply by dropping them and using cd
+ pwd
enclosed in a command substitution instead. SCRIPT_DIR=$(...)
Incredible how simple can be, no matter how you call the script:
#!/bin/bash
#
the_source=$(readlink -f -- "${BASH_SOURCE[0]}")
the_dirname=$(dirname "${the_source}")
echo "the_source: ${the_source}"
echo "the_dirname: ${the_dirname}"
Run from anywhere:
user@computer:~/Downloads/temp$ ./test.sh
Output:
the_source: /home/user/Downloads/temp/test.sh
the_dirname: /home/user/Downloads/temp
2 Comments
.bin
directory created by npm/yarn
and you need to now original script location to relatively reference static resources bundled in your npm package...--
: readlink -f -- "${BASH_SOURCE[0]}"
to avoid an error if the script name starts with -
. 2. BSD/macOS readlink does not support -f
. 3. Add || exit $?
or || exit 1
to both lines to avoid swallowing errors.I would use something like this:
# Retrieve the full pathname of the called script
scriptPath=$(which 0ドル)
# Check whether the path is a link or not
if [ -L $scriptPath ]; then
# It is a link then retrieve the target path and get the directory name
sourceDir=$(dirname $(readlink -f $scriptPath))
else
# Otherwise just get the directory name of the script path
sourceDir=$(dirname $scriptPath)
fi
2 Comments
sh
too! Problem with simple dirname "0ドル"
based solutions: If the script is in the $PATH
and is invoked without path, they will give wrong result.PATH
, 0ドル
will contain the absolute filename. If the script is invoked with a relative or absolute filename containing a /
, 0ドル
will contain that.For systems having GNU coreutils readlink
(for example, Linux):
$(readlink -f "$(dirname "0ドル")")
There's no need to use BASH_SOURCE
when 0ドル
contains the script filename.
1 Comment
$_
is worth mentioning as an alternative to 0ドル
. If you're running a script from Bash, the accepted answer can be shortened to:
DIR="$( dirname "$_" )"
Note that this has to be the first statement in your script.
2 Comments
source
or .
the script. In those situations, $_
would contain the last parameter of the last command you ran before the .
. $BASH_SOURCE
works every time.These are short ways to get script information:
Folders and files:
Script: "/tmp/src dir/test.sh"
Calling folder: "/tmp/src dir/other"
Using these commands:
echo Script-Dir : `dirname "$(realpath 0ドル)"`
echo Script-Dir : $( cd ${0%/*} && pwd -P )
echo Script-Dir : $(dirname "$(readlink -f "0ドル")")
echo
echo Script-Name : `basename "$(realpath 0ドル)"`
echo Script-Name : `basename 0ドル`
echo
echo Script-Dir-Relative : `dirname "$BASH_SOURCE"`
echo Script-Dir-Relative : `dirname 0ドル`
echo
echo Calling-Dir : `pwd`
And I got this output:
Script-Dir : /tmp/src dir
Script-Dir : /tmp/src dir
Script-Dir : /tmp/src dir
Script-Name : test.sh
Script-Name : test.sh
Script-Dir-Relative : ..
Script-Dir-Relative : ..
Calling-Dir : /tmp/src dir/other
Also see: https://pastebin.com/J8KjxrPF
Comments
This works in Bash 3.2:
path="$( dirname "$( which "0ドル" )" )"
If you have a ~/bin
directory in your $PATH
, you have A
inside this directory. It sources the script ~/bin/lib/B
. You know where the included script is relative to the original one, in the lib
subdirectory, but not where it is relative to the user's current directory.
This is solved by the following (inside A
):
source "$( dirname "$( which "0ドル" )" )/lib/B"
It doesn't matter where the user is or how he/she calls the script. This will always work.
2 Comments
which
is very debatable. type
, hash
, and other builtins do the same thing better in bash. which
is kindof more portable, though it really isn't the same which
used in other shells like tcsh, that has it as a builtin.which
being an external tool, you have no reason to believe it behaves identically to the parent shell.I've compared many of the answers given, and came up with some more compact solutions. These seem to handle all of the crazy edge cases that arise from your favorite combination of:
- Absolute paths or relative paths
- File and directory soft links
- Invocation as
script
,bash script
,bash -c script
,source script
, or. script
- Spaces, tabs, newlines, Unicode, etc. in directories and/or filename
- Filenames beginning with a hyphen
If you're running from Linux, it seems that using the proc
handle is the best solution to locate the fully resolved source of the currently running script (in an interactive session, the link points to the respective /dev/pts/X
):
resolved="$(readlink /proc/$$/fd/255 && echo X)" && resolved="${resolved%$'\nX'}"
This has a small bit of ugliness to it, but the fix is compact and easy to understand. We aren't using bash primitives only, but I'm okay with that because readlink
simplifies the task considerably. The echo X
adds an X
to the end of the variable string so that any trailing whitespace in the filename doesn't get eaten, and the parameter substitution ${VAR%X}
at the end of the line gets rid of the X
. Because readlink
adds a newline of its own (which would normally be eaten in the command substitution if not for our previous trickery), we have to get rid of that, too. This is most easily accomplished using the $''
quoting scheme, which lets us use escape sequences such as \n
to represent newlines (this is also how you can easily make deviously named directories and files).
The above should cover your needs for locating the currently running script on Linux, but if you don't have the proc
filesystem at your disposal, or if you're trying to locate the fully resolved path of some other file, then maybe you'll find the below code helpful. It's only a slight modification from the above one-liner. If you're playing around with strange directory/filenames, checking the output with both ls
and readlink
is informative, as ls
will output "simplified" paths, substituting ?
for things like newlines.
absolute_path=$(readlink -e -- "${BASH_SOURCE[0]}" && echo x) && absolute_path=${absolute_path%?x}
dir=$(dirname -- "$absolute_path" && echo x) && dir=${dir%?x}
file=$(basename -- "$absolute_path" && echo x) && file=${file%?x}
ls -l -- "$dir/$file"
printf '$absolute_path: "%s"\n' "$absolute_path"
4 Comments
/dev/pts/30
with bash on Ubuntu 14.10 Desktop.echo $resolved
, I saved it as d
, chmod +x d
, ./d
.#!/bin/bash
Try the following cross-compatible solution:
CWD="$(cd -P -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd -P)"
As the commands such as realpath
or readlink
could be not available (depending on the operating system).
Note: In Bash, it's recommended to use ${BASH_SOURCE[0]}
instead of 0ドル
, otherwise path can break when sourcing the file (source
/.
).
Alternatively you can try the following function in Bash:
realpath () {
[[ 1ドル = /* ]] && echo "1ドル" || echo "$PWD/${1#./}"
}
This function takes one argument. If argument has already absolute path, print it as it is, otherwise print $PWD
variable + filename argument (without ./
prefix).
Related:
2 Comments
realpath
function takes 1 argument. If argument has already absolute path, print it as it is, otherwise print $PWD
+ filename (without ./
prefix).
DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd && echo x)"
- and remove it without a command substitution -DIR="${DIR%x}"
.mkdir $'\n'
.