Check If Database Exists In PostgreSQL Using Shell


Answer :

Note/Update (2021): While this answer works, philosophically I agree with other comments that the right way to do this is to ask Postgres.

Check whether the other answers that have psql -c or --command in them are a better fit for your use case (e.g. Nicholas Grilly's, Nathan Osman's, bruce's or Pedro's variant


I use the following modification of Arturo's solution:

psql -lqt | cut -d \| -f 1 | grep -qw <db_name>


What it does

psql -l outputs something like the following:

                                        List of databases      Name  |   Owner   | Encoding |  Collate   |   Ctype    |   Access privileges    -----------+-----------+----------+------------+------------+-----------------------  my_db     | my_user   | UTF8     | en_US.UTF8 | en_US.UTF8 |   postgres  | postgres  | LATIN1   | en_US      | en_US      |   template0 | postgres  | LATIN1   | en_US      | en_US      | =c/postgres          +            |           |          |            |            | postgres=CTc/postgres  template1 | postgres  | LATIN1   | en_US      | en_US      | =c/postgres          +            |           |          |            |            | postgres=CTc/postgres (4 rows) 

Using the naive approach means that searching for a database called "List, "Access" or "rows" will succeed. So we pipe this output through a bunch of built-in command line tools to only search in the first column.


The -t flag removes headers and footers:

 my_db     | my_user   | UTF8     | en_US.UTF8 | en_US.UTF8 |   postgres  | postgres  | LATIN1   | en_US      | en_US      |   template0 | postgres  | LATIN1   | en_US      | en_US      | =c/postgres          +            |           |          |            |            | postgres=CTc/postgres  template1 | postgres  | LATIN1   | en_US      | en_US      | =c/postgres          +            |           |          |            |            | postgres=CTc/postgres 

The next bit, cut -d \| -f 1 splits the output by the vertical pipe | character (escaped from the shell with a backslash), and selects field 1. This leaves:

 my_db               postgres            template0                               template1                    

grep -w matches whole words, and so won't match if you are searching for temp in this scenario. The -q option suppresses any output written to the screen, so if you want to run this interactively at a command prompt you may with to exclude the -q so something gets displayed immediately.

Note that grep -w matches alphanumeric, digits and the underscore, which is exactly the set of characters allowed in unquoted database names in postgresql (hyphens are not legal in unquoted identifiers). If you are using other characters, grep -w won't work for you.


The exit status of this whole pipeline will be 0 (success) if the database exists or 1 (failure) if it doesn't. Your shell will set the special variable $? to the exit status of the last command. You can also test the status directly in a conditional:

if psql -lqt | cut -d \| -f 1 | grep -qw <db_name>; then     # database exists     # $? is 0 else     # ruh-roh     # $? is 1 fi 

The following shell code seems to work for me:

if [ "$( psql -tAc "SELECT 1 FROM pg_database WHERE datname='DB_NAME'" )" = '1' ] then     echo "Database already exists" else     echo "Database does not exist" fi 

postgres@desktop:~$ psql -l | grep <exact_dbname> | wc -l 

This will return 1 if the database specified exists or 0 otherwise.

Also, if you try to create a database that already exists, postgresql will return an error message like this:

postgres@desktop:~$ createdb template1 createdb: database creation failed: ERROR:  database "template1" already exists 

Comments

Popular posts from this blog

Chemistry - Bond Angles In NH3 And NCl3

Are Regular VACUUM ANALYZE Still Recommended Under 9.1?

Change The Font Size Of Visual Studio Solution Explorer