= Use the [ command on Linux The `[` command, often called a "test," is a command from the GNU Core Utils package, and initiates a conditional statement in Bash. Its function is exactly the same as the `test` command. When you want to execute a command only when something is either true or false, use the `[` or the `test` command. However, there's a significant difference between `[` or `test` and `[[`, and there's a technical difference between those commands and your shell's versions of them. == Which command is which The `[` and the `test` commands, installed by the GNU Core Utils package, perform the same function using a slightly different syntax. (You might find it difficult to search for documentation using the single left-square bracket character, however, so many users find `test` easier to reference.) Bash and similar shells happen to _also_ have the `[` and the `test` commands built-in, and the built-in versions supersede the ones installed in `/usr/bin`. In other words, when you use `[` or `test`, you're probably not executing `/usr/bin/[` or `/usr/bin/test`. Instead, you're invoking what's essentially a function of your Bash shell. You might wonder why `[` or `test` exist in `/usr/bin` at all. Some shells, such as https://opensource.com/article/20/8/tcsh[tcsh], don't have `[` and `test` built-in, so if you want to use those commands in that shell, you must have them installed as separate binaries. The bottom line is that as long as you don't get an error when you type a command starting with `[` or `test`, then you've got everything you need. It almost never matters whether your shell or your `bin` directory is providing the commands. == Testing for a file It's common to want to know whether a file exists, often so you can confidently proceed with some action, or so you can avoid "clobbering" it with a file of the same name. In an interactive shell session, you can just look to see whether the file exists but in a shell script, you need the computer to determine that for itself. The `-e` option tests whether a file exists, but its apparent response is the same either way. [source,bash] ---- $ touch example $ test -e example $ test -e notafile $ ---- The `[` and `test` commands are essentially switches. They emit a `true` or `false` response, but considers both of them as success. You can put this to use by pairing the commands with logical operators, such as `&&` and `||`. The `&&` operator is executes when a response is `true`: [source,bash] ---- $ touch example $ test -e example && echo "foo" foo $ test -e notafile && echo "foo" $ ---- The `||` operator executes when a response is `false`: [source,bash] ---- $ touch example $ test -e example || echo "foo" $ test -e notafile || echo "foo" foo $ ---- If you prefer, you can use square brackets instead of `test`. In all cases, the results are the same: [source,bash] ---- $ touch example $ [ -e example ] && echo "foo" foo $ [ -e notafile ] && echo "foo" $ ---- == Testing for file types Everything in Linux is a file, so when you can test for the existence of a directory with the `-e` option, the same way you test for a file. However, there are different kinds of files, and sometimes that matters. You can use `[` or `test` to detect a variety of different file types: * `-f`: regular file (returns `false` for a directory) * `-d`: directory * `-b`: block (such as `/dev/sda1`) * `-L` or `-h`: symlink * `-S`: socket There are more, but those tend to be the most common. == Testing for file attributes You can also look at metadata of a file: * `-s`: a file with the size greater than zero * `-N`: a file that's been modified since it was last read You can test by ownership: * `-O`: a file owned by the current primary user * `-G`: a file owned by the current primary group Or you can test by permissions (or _file mode_): * `-r`: a file with read permission granted * `-w`: a file with write permission granted * `-x`: a file with execute permission granted * `-k`: a file with the sticky bit set == Combining tests You don't always just have to test for a single attribute. The `-a` option ("and") allows you to string several tests together, with the requirement that all tests return as `true`: [source,bash] ---- $ touch zombie apocalypse now $ test -e zombie -a -e apocalypse -a -e now && echo "no thanks" no thanks ---- If any expression fails, then the test returns `false`: [source,bash] ---- $ touch zombie apocalypse now $ test -e plant -a -e apocalypse -a -e now && echo "no thanks" $ ---- The `-o` option ("or") requires that one expression is true: [source,bash] ---- $ touch zombie apocalypse now $ test -e zombie -o -e plant -o -e apocalypse && echo "no thanks" no thanks ---- == Integer tests You can also test integers. That's not necessarily directly useful (you probably inherently know that 0 is less than 1, for instance) but it's invaluable when you're using variables in a script. The operators are fairly intuitive once you understand the schema: * `-eq`: equal to * `-ne`: not equal * `-ge`: greater than or equal to * `-gt`: greater than * `-le`: less than or equal to * `-lt`: less than Here's a simple example: [source,bash] ---- $ nil=0 $ foo=1 $ test $foo -eq $nil || echo "Those are not equal." Those are not equal. $ test $foo -eq 1 && echo "Those are equal." ---- Of course, you can combine tests. [source,bash] ---- $ touch example $ test $foo -ne $nil -a -e example -o -e notafile && echo "yes" yes ---- == Testing testing The `[` and `test` commands are vital conditional statements when scripting. These are easy and common ways to control the flow of your code. There are yet more tests available than what I've covered in this article, so whether you used Bash, tcsh, ksh, or some other shell entirely, take a look at the man page to get the full spectrum of what these commands offer.