A Must Have PostgreSQL Notes For Professionals 1688349123
A Must Have PostgreSQL Notes For Professionals 1688349123
About ................................................................................................................................................................................... 1
Chapter 1: Getting started with PostgreSQL .................................................................................................. 2
Section 1.1: Installing PostgreSQL on Windows ........................................................................................................... 2
Section 1.2: Install PostgreSQL from Source on Linux ............................................................................................... 3
Section 1.3: Installation on GNU+Linux ......................................................................................................................... 4
Section 1.4: How to install PostgreSQL via MacPorts on OSX ................................................................................... 5
Section 1.5: Install postgresql with brew on Mac ........................................................................................................ 7
Section 1.6: Postgres.app for Mac OSX ........................................................................................................................ 7
Chapter 17: Export PostgreSQL database table header and data to CSV file ........................... 38
Section 17.1: copy from query .................................................................................................................................... 38
Section 17.2: Export PostgreSQL table to csv with header for some column(s) ................................................... 38
Section 17.3: Full table backup to csv with header ................................................................................................... 38
Credits .............................................................................................................................................................................. 68
You may also like ........................................................................................................................................................ 70
About
This PostgreSQL® Notes for Professionals book is compiled from Stack Overflow
Documentation, the content is written by the beautiful people at Stack
Overflow.
Text content is released under Creative Commons BY-SA, see
credits at the end of this book whom contributed to the
various chapters. Images may be copyright of their respective
owners unless otherwise specified
Select the latest stable (non-Beta) version (9.5.3 at the time of writing). You will most likely want
the Win x86-64 package, but if you are running a 32 bit version of Windows, which is common on
older computers, select Win x86-32 instead.
Note: Switching between Beta and Stable versions will involve complex tasks like dump and
restore. Upgrading within beta or stable version only needs a service restart.
You can check if your version of Windows is 32 or 64 bit by going to Control Panel -> System and
Security -> System -> System type, which will say "##-bit Operating System". This is the path for
Windows 7, it may be slightly different on other versions of Windows.
In the installer select the packages you would like to use. For example:
All those optional packages can be later installed through "Application Stack Builder".
Note: There are also other non-officially supported language such as PL/V8, PL/Lua PL/Java
available.
Open pgAdmin and connect to your server by double clicking on its name, ex. "PostgreSQL 9.5
(localhost:5432).
From this point you can follow guides such as the excellent book PostgreSQL: Up and Running,
2nd Edition ( http://shop.oreilly.com/product/0636920032144.do ).
Optional: Manual Service Startup Type
PostgreSQL runs as a service in the background which is slightly different than most programs.
This is common for databases and web servers. Its default Startup Type is Automatic which
means it will always run without any input from you.
Why would you want to manually control the PostgreSQL service? If you're using your PC as a
development server some of the time and but also use it to play video games for example,
PostegreSQL could slow down your system a bit while its running.
Why wouldn't you want manual control? Starting and stopping the service can be a hassle if you
do it often.
If you don't notice any difference in speed and prefer avoiding the hassle then leave its Startup
Type as Automatic and ignore the rest of this guide. Otherwise...
Select "Services" from the list, right click on its icon, and select Send To -> Desktop to create a
desktop icon for more convenient access.
Close the Administrative Tools window then launch Services from the desktop icon you just
created.
Scroll down until you see a service with a name like postgresql-x##-9.# (ex. "postgresql-x64-
9.5").
Right click on the postgres service, select Properties -> Startup type -> Manual -> Apply -> OK.
You can change it back to automatic just as easily.
If you see other PostgreSQL related services in the list such "pgbouncer" or "PostgreSQL
Scheduling Agent pgAgent" you can also change their Startup Type to Manual because they're
not much use if PostgreSQL isn't running. Although this will mean more hassle each time you
start and stop so it's up to you. They don't use as many resources as PostgreSQL itself and may
not have any noticeable impact on your systems performance.
If the service is running its Status will say Started, otherwise it isn't running.
To start it right click and select Start. A loading prompt will be displayed and should disappear
on its own soon after. If it gives you an error try a second time. If that doesn't work then there
was some problem with the installation, possibly because you changed some setting in Windows
most people don't change, so finding the problem might require some sleuthing.
If you ever get an error while attempting to connect to your database check Services to make
sure its running.
For other very specific details about the EDB PostgreSQL installation, e.g. the python runtime
version in the official language pack of a specific PostgreSQL version, always refer to the official
EBD installation guide , change the version in link to your installer's major version.
libedit-devel Sources:
There are a large number of different options for the configuration of PostgreSQL:
installation procedure
options:
Go into the new created folder and run the cofigure script with the desired options:
./configure -- exec=/usr/local/pgsql
For the extension switch the directory cd contrib , run make and make
install
These are installed with the following command: yum -y install postgresqlXX postgresqlXX-server
postgresqlXX-libs postgresqlXX-contrib
Once installed you will need to start the database service as the service owner (Default is
postgres). This is done with the pg_ctl command.
sudo - su postgres
. / usr / pgsql-X.X / bin / pg_ctl - D / var / lib / pgsql / X.X / data start
Debian family
This will install the PostgreSQL server package, at the default version offered by the operating
system's package repositories.
If the version that's installed by default is not the one that you want, you can use the package
manager to search for specific versions which may simultaneously be offered.
You can also use the Yum repository provided by the PostgreSQL project (known as PGDG)
to get a different version. This may allow versions not yet offered by operating system
package repositories.
You should get a list that looks something like the following:
postgresql80 @8.0.26
databases/postgresql80 postgresql81 @8.1.23
databases/postgresql81 postgresql82 @8.2.23
databases/postgresql82 postgresql83 @8.3.23
databases/postgresql83 postgresql84 @8.4.22
databases/postgresql84 postgresql90 @9.0.23
databases/postgresql90 postgresql91 @9.1.22
databases/postgresql91 postgresql92 @9.2.17
databases/postgresql92 postgresql93 @9.3.13
databases/postgresql93 postgresql94 @9.4.8
databases/postgresql94 postgresql95 @9.5.3
databases/postgresql95 postgresql96 @9.6beta2
databases/postgresql96
In this example, the most recent version of PostgreSQL that is supported in 9.6, so we will install
that.
sudo port install postgresql96-server postgresql96
server port
The log provides instructions on the rest of the steps for installation, so we do that next.
psql (9.6.1)
Type "help" for help.
postgres=#
Here you can type a query to see that the server is running.
Type \q to quit:
postgres = #\q
Homebrew generally installs the latest stable version. If you need a different one then brew SEARCH postgresql
will
list the versions available. If you need PostgreSQL built with particular options then brew info
postgresql will list which options are supported. If you require an unsupported build option, you
may have to do the build yourself, but can still use Homebrew to install the common
dependencies.
If psql complains that there's no corresponding database for your user, run
CREATEDB.
https://www.postgresql.org/docs/9.6/static/datatype.html
DOUBLE PRECISION
Name Storage Size Description Range
SMALLINT 2 bytes small-range integer -32768 to +32767
INTEGER 4 bytes ypical choice for integer -2147483648 to +2147483647
-9223372036854775808 to
BIGINT 8 bytes large-range integer
+9223372036854775807
up to 131072 digits before the decimal point; up
DECIMAL variable user-specified precision, exact
to 16383 digits after the decimal point
up to 131072 digits before the decimal point; up
NUMERIC variable user-specified precision, exact
to 16383 digits after the decimal point
REAL 4 bytes variable-precision, inexact 6 decimal digits precision
8 bytes variable-precision, inexact 15 decimal digits precision
smallserial 2 bytes small autoincrementing integer 1 to 32767 serial 4
bytes autoincrementing integer 1 to 2147483647
BIGSERIAL 8 bytes large autoincrementing integer 1 to 9223372036854775807
int4range Range of integer int8range Range of
bigint numrange Range of numeric
Creating an Array
SELECT '{0,1,2}' ;
SELECT '{{0,1},{1,2}}' ;
SELECT ARRAY[ 0 , 1 , 2 ] ;
SELECT ARRAY[ ARRAY[ 0 , 1 ] , ARRAY[ 1 , 2 ] ] ;
Accessing an Array
By default PostgreSQL uses a one-based numbering convention for arrays, that is, an array of n
ARRAY[1] and ends with ARRAY elements starts with [n].
int_arr
---------
0
( 1 ROW)
-- sclicing an array
WITH arr AS ( SELECT ARRAY[ 0 , 1 , 2 ] int_arr ) SELECT int_arr [ 1 : 2 ] FROMarr;
int_arr
---------
{ 0, 1}
( 1 ROW)
Getting information about an array
-- array dimensions (as text )
WITH arr AS ( SELECT ARRAY[ 0 , 1 , 2 ] int_arr ) SELECT ARRAY_DIMS( int_arr ) FROMarr;
array_dims
------------
[ 1: 3]
( 1 ROW)
array_length
--------------
3
( 1 ROW)
cardinality
-------------
3
( 1 ROW)
Array functions
will be added
This statement will produce the string "12 Aug 2016 04:40:32PM". The formatting string can be
modified in many different ways; the full list of template patterns can be found here.
Note that you can also insert plain text into the formatting string and you can use the template
patterns in any order:
This will produce the string "Today is Saturday, the 12th day of the month of August of 2016".
You should keep in mind, though, that any template patterns - even the single letter ones like
"I", "D", "W" - are converted, unless the plain text is in double quotes. As a safety measure, you
should put all plain text in double quotes, as done above.
You can localize the string to your language of choice (day and month names) by using the TM
(translation mode) modifier. This option uses the localization setting of the server running
PostgreSQL or the client connecting to it.
With a Spanish locale setting this produces "Sábado, 12 de Agosto del año 2016".
If you have forgotten the name of the table, just type \d into psql to obtain a list of tables and
views in the current database. Section 4.2: Create table from select
Let's say you have a table called person:
CREATE TABLE person (
person_id BIGINT NOT NULL,
last_name VARCHAR ( 255 ) NOT NULL,
first_name VARCHAR ( 255 ) ,
age INT NOT NULL,
PRIMARY KEY ( person_id )
);
You can create a new table of people over 30 like this:
CREATE TABLE people_over_30 AS SELECT * FROM person WHERE age > 30;
);
Alternatively, you can place the PRIMARY KEY constraint directly in the column definition:
CREATE TABLE person (
person_id BIGINT NOT NULL PRIMARY KEY,
last_name VARCHAR ( 255 ) NOT NULL,
first_name VARCHAR ( 255 ) ,
address VARCHAR ( 255 ) ,
city VARCHAR ( 255 )
);
It is recommended that you use lower case names for the table and as well as all the columns.
If you use upper case names such as Person you would have to wrap that name in double
quotes ("Person") in each and every query because PostgreSQL enforces case folding.
Chapter 5: SELECT
Section 5.1: SELECT using WHERE
In this topic we will base on this table of users :
CREATE TABLE sch_test.user_table
(
id serial NOT NULL,
username CHARACTER VARYING,
pass CHARACTER VARYING,
first_name CHARACTERvarying ( 30 ) ,
last_name CHARACTERvarying ( 30 ) ,
CONSTRAINT user_table_pkey PRIMARY KEY ( id )
)
+----+------------+-----------+----------+------+
| id | first_name | last_name | username | pass |
+----+------------+-----------+----------+------+
| 1 | hello | world | hello | word |
+----+------------+-----------+----------+------+
| 2 | root | me | root | toor |
+----+------------+-----------+----------+------+
Syntax
Examples
-- SELECT every thing where id = 1
SELECT * FROMschema_name.table_name WHEREid = 1;
Result:
Result:
5
Chapter 7: COALESCE
Coalesce returns the first none null argument from a set of arguments. Only the first non null
argument is return, all subsequent arguments are ignored. The function will evaluate to null if all
arguments are null.
COALESCE
--------
'HELLO WORLD'
coalesce
--------
'first non null'
COALESCE
--------
Chapter 8: INSERT
Section 8.1: Insert data using COPY
COPY is PostgreSQL's bulk-insert mechanism. It's a convenient way to transfer data between
files and tables, but it's also far faster than INSERT when adding more than a few thousand rows
at a time.
1,Yogesh
2,Raunak
3,Varun
4,Kamal
5,Hari
6,Amit
And we need a two column table into which this data can be imported into.
CREATE TABLE copy_test ( id INT , NAMEvarchar ( 8) ) ;
Now the actual copy operation, this will create six records in the table.
SELECT * FROMcopy_test ;
id | name
----+--------
1 | Yogesh
3 | Varun
5 | Hari
7 | Amol
2 | Raunak
4 | Kamal
6 | Amit
8 | Amar
Above query will return the id of the row where the new record was
The most basic insert involves inserting all values in the table:
INSERT INTO person VALUES ( 1 , 'john doe' , 25 , 'new york' );
If you want to insert only specific columns, you need to explicitly indicate which columns:
INSERT INTO person ( NAME, age ) VALUES ( 'john doe' , 25 ) ;
Note that if any constraints exist on the table , such as NOT NULL, you will be required to include
Note that the projection of the select must match the columns required for the insert. In this
case, the tmp_person table has the same columns as person. Section 8.6: UPSERT -
INSERT ... ON CONFLICT DO UPDATE..
since version 9.5 postgres offers UPSERT functionality with INSERT statement.
Say you have a table called my_table, created in several previous examples. We insert a row,
returning PK value of inserted row:
INSERT 0 1
Now if we try to insert row with existing unique key it will raise an exception:
b=# INSERT INTO my_table VALUES (2,'one',333); ERROR: duplicate KEY
VALUE violates UNIQUE CONSTRAINT "my_table_pkey" DETAIL: KEY
(id)=(2) already EXISTS.
b=# INSERT INTO my_table values (2,'one',333) ON CONFLICT (id) DO UPDATE SET name
= my_table.name||' changed to: "two" at '||now() returning *; id |
name | contact_number ----+---------------------------
------------------------------------------------------------------
--------------+----------------
2 | one changed to: "two" at 2016-11-23 08:32:17.105179+00 | 333
(1 row)
INSERT 0 1
Here we are joining the person city column to the cities city column in order to get the city's state
code. This is then used to update the state_code column in the person table. Section 9.2:
Update all rows in a table
You update all rows in table by simply providing a column_name = VALUE :
UPDATE person SET planet = 'Earth' ;
operator returns the value of JSON Column. Section 10.1: Using JSONb
operators
Creating a DB and a Table
DROP DATABASE IF EXISTS books_db;
CREATE DATABASE books_db WITH ENCODING= 'UTF8' TEMPLATEtemplate0;
Populating the DB
INSERT INTO books ( client, DATA) VALUES (
'Joe' ,
'{ "title": "Siddhartha", "author": { "first_name": "Herman", "last_name": "Hesse" } }'
), (
'Jenny' ,
'{ "title": "Dharma Bums", "author": { "first_name": "Jack", "last_name": "Kerouac" } }'
), (
'Jenny' ,
'{ "title": "100 a ños de soledad", "author": { "first_name": "Gabo", "last_name": "Marqu éz" }
}'
);
Output:
SELECT client,
DATA- > 'title' AS title
FROMbooks;
Output:
Selecting 2 columns:
SELECT client,
DATA- > 'title' AS title, DATA- > 'author' AS author
FROMbooks;
Output:
-> vs - >>
The -> operator returns the original JSON type (which might be an object), whereas - >>
You can use the -> to return a nested object and thus chain the operators:
SELECT client,
DATA- > 'author' - > 'last_name' AS author
FROMbooks;
Output:
Filtering
Output:
Nested filtering
Output:
We’re going to store events in this table, like pageviews. Each event has properties, which could
be anything (e.g. current page) and also sends information about the browser (like OS, screen
resolution, etc). Both of these are completely free form and could change over time (as we think
of extra stuff to track).
INSERT INTO events ( NAME, visitor_id, properties, browser ) VALUES
(
'pageview' , '1' ,
'{ "page": "/" }' ,
'{ "name": "Chrome", "os": "Mac", "resolution": { "x": 1440, "y": 900 } }'
), (
'pageview' , '2' ,
'{ "page": "/" }' ,
'{ "name": "Firefox", "os": "Windows", "resolution": { "x": 1920, "y": 1200 } }'
), (
'pageview' , '1' ,
'{ "page": "/account" }' ,
'{ "name": "Chrome", "os": "Mac", "resolution": { "x": 1440, "y": 900 } }'
), (
'purchase' , '5' ,
'{ "amount": 10 }' ,
'{ "name": "Firefox", "os": "Windows", "resolution": { "x": 1024, "y": 768 } }'
), (
'purchase' , '15' ,
'{ "amount": 200 }' ,
'{ "name": "Firefox", "os": "Windows", "resolution": { "x": 1280, "y": 800 } }'
), (
'purchase' , '15' ,
'{ "amount": 500 }' ,
'{ "name": "Firefox", "os": "Windows", "resolution": { "x": 1280, "y": 800 } }'
);
Now lets select everything:
SELECT * FROMevents;
Output:
Using the JSON operators, combined with traditional PostgreSQL aggregate functions, we can
pull out whatever we want. You have the full might of an RDBMS at your disposal.
Lets see browser usage:
Output:
Output:
SELECT AVG( CAST( browser- > 'resolution' - >> 'x' AS INTEGER ) ) AS width,
AVG( CAST( browser- > 'resolution' - >> 'y' AS INTEGER ) ) AS height
FROMevents;
Output:
More examples and documentation here and here.
It is important to understand the performance difference between using @>, -> and - >> in the
WHERE part of the query. Although these two queries appear to be broadly equivalent:
created above whereas the latter two will not, requiring a complete table scan.
It is still allowable to use the -> operator when obtaining resultant data, so the following queries
will also use the index:
At this point you can insert data in to the table and query it efficiently.
Name Age
Allie 17
Amanda 14
Alana 20
You could write this statement to get the minimum, maximum and average value:
SELECT MIN( age ) , MAX( age ) , AVG( age )
FROMindividuals;
Result:
min max avg
14 20 17
All memory leak candidates will have a trend of consuming more memory as more time passes.
If you plot this trend, you would imagine a line going up and to the left:
^
|
s | Legend:
i | * - data point
z | -- - trend
e |
( |
b | *
y | --
t | --
e | * -- *
s | --
) | *-- *
| -- *
| -- *
--------------------------------------->
time
Suppose you have a table containing heap dump histogram data (a mapping of classes to how
much memory they consume):
CREATE TABLE heap_histogram (
-- when the heap histogram was taken
histwhen TIMESTAMP WITHOUT TIME ZONE NOT NULL,
-- the object type bytes are referring to
-- ex: java.util.String
CLASS CHARACTER VARYING NOT NULL,
-- the size in bytes used by the above class
bytes INTEGER NOT NULL
);
To compute the slope for each class, we group by over the class. The HAVING clause > 0 ensures
that we get only candidates with a positive slop (a line going up and to the left). We sort by the
slope descending so that we get the classes with the largest rate of memory increase at the top.
-- epoch returns seconds
SELECT CLASS, REGR_SLOPE( bytes, EXTRACT( epoch FROMhistwhen ) ) AS slope
FROMpublic.heap_histogram
GROUP BY CLASS
HAVING REGR_SLOPE( bytes, EXTRACT( epoch FROMhistwhen ) ) > 0
ORDER BY slope DESC ;
Output:
class | slope
---------------------------+----------------------
java.util.ArrayList | 71.7993806279174
java.util.HashMap | 49.0324576155785
java.lang.String | 31.7770770326123
joe.schmoe.BusinessObject | 23.2036817108056
java.lang.ThreadLocal | 20.9013528767851
From the output we see that java.util.ArrayList's memory consumption is increasing the fastest
at 71.799 bytes per second and is potentially part of the memory leak. Section 11.3:
string_agg(expression, delimiter)
You can concatenate strings separated by delimiter using the STRING_AGG () function.
You could write SELECT ... GROUP BY statement to get names from each country:
SELECT STRING_AGG( NAME, ', ' ) AS NAMES, country
FROMindividuals
GROUP BY country;
Note that you need to use a GROUP BY clause because STRING_AGG () is an aggregate function.
Result:
names country
Allie, Amanda USA
Alana Russia
Running:
SELECT *
, DENSE_RANK( ) OVER ( ORDER BY i ) dist_by_i
, LAG( t ) OVER ( ) prev_t
, NTH_VALUE( i, 6 ) OVER ( ) nth
, COUNT( TRUE) OVER ( PARTITION BY i ) num_by_i
, COUNT( TRUE) OVER ( ) num_all
, NTILE ( 3 ) over ( ) ntile
FROMwf_example
;
Result:
Explanation:
dist_by_i: DENSE_RANK() OVER ( ORDER BY i ) is like a row_number per distinct values. Can be
used for the number COUNT( DISTINCT i of distinct values of i () wold not work). Just use the
maximum value.
prev_t: LAG(t) OVER () is a previous value of t over the whole window. mind that it is null for the
first row.
nth: NTH_VALUE( i, 6) OVER () is the value of sixth rows column i over the
ntile: NTILE (3) over () splits the whole window to 3 (as much as possible) equal in quantity parts
Section 13.2: column values vs dense_rank vs rank vs
row_number
here you can find the functions.
dense_rank orders VALUES of i by appearance in window. i=1 appears, so first row has
dense_rank, next and third i value does not change, so it is dense_rank shows 1 - FIRST
value not changed. fourth row i=2, it is second value of i met, so dense_rank shows 2, andso
for the next row. Then it meets value i=3 at 6th row, so it show 3. Same for the rest two
values of i. So the last value of dense_rank is the number of distinct values of i.
rank Not to confuse with dense_rank this function orders ROW NUMBER of i values. So it
starts same with three ones, but has next value 4, which means i=2 (new value) was met at
row 4. Same i=3 was met at row 6.
Etc..
Link to Documentation
Chapter 15: Programming with PL/pgSQL
Section 15.1: Basic PL/pgSQL Function
A simple PL/pgSQL function:
CREATE FUNCTION active_subscribers ( ) RETURNS BIGINT AS $$
DECLARE
-- variable for the following BEGIN ... END block
subscribers INTEGER ;
BEGIN
-- SELECT must always be used with INTO
SELECT COUNT( user_id ) INTO subscribers FROMusers WHEREsubscribed;
-- function result
RETURNsubscribers;
EXCEPTION
-- return NULL if table "users" does not exist
WHENundefined_table
THEN RETURN NULL;
END;
$$ LANGUAGEplpgsql;
This could have been achieved with just the SQL statement but demonstrates the basic structure
of a function.
calling:
t = # DO
$$
DECLARE
_t TEXT;
BEGIN
perform s165 ();
exception WHENSQLSTATE 'P0001' THEN raise info '%' , 'state P0001 caught: ' || SQLERRM;
perform s164 ();
END;
$$
;
INFO: state P0001 caught: NOTHING specified
ERROR: S 164
DETAIL: D 164
HINT: H 164
CONTEXT: SQL STATEMENT "SELECT s164()"
PL/pgSQL FUNCTION inline_code_block line 7 AT PERFORM
here custom P0001 processed, and P2222, not, aborting the execution.
Chapter 16:
Inheritance
Section 16.1: Creating children tables
CREATE TABLE users (username TEXT, email TEXT);
CREATE TABLE simple_users () INHERITS (users);
CREATE TABLE users_with_password (PASSWORD TEXT) INHERITS (users);
users
C
o
l
u
m
n
T
y
p
e
u
s
e
r
n
a
m
e
t
e
x
t
e
m
a
il
t
e
x
t
simple_users
C
o
l
u
m
n
T
y
p
e
u
s
e
r
n
a
m
e
t
e
x
t
e
m
a
il
t
e
x
t
users_with_password
C
o
l
u
m
n
T
y
p
e
u
s
e
r
n
a
m
e
t
e
x
t
e
m
a
il
t
e
x
t
p
a
s
s
w
o
r
d
t
e
x
t
Chapter 17: Export PostgreSQL database
table header and data to CSV file
From Adminer management tool it's has export to csv file option for mysql database But not
available for postgresql database. Here I will show the command to export CSV for postgresql
database.
Step 3: test it
INSERT INTO company ( NAME) VALUES ( 'My company' );
RETURN vReturn;
END $BODY$
LANGUAGEplpgsql;
Step 3: test it
INSERT INTO company ( NAME) VALUES ( 'Company 1' ) ;
INSERT INTO company ( NAME) VALUES ( 'Company 2' ) ;
INSERT INTO company ( NAME) VALUES ( 'Company 3' ) ;
UPDATE company SET NAME= 'Company new 2' WHERE NAME= 'Company 2' ;
DELETE FROMcompany WHERE NAME= 'Company 1' ;
SELECT * FROMlog;
Section 18.2: Basic PL/pgSQL Trigger Function
This is a simple trigger function.
CREATE OR REPLACE FUNCTION my_simple_trigger_function ()
RETURNS TRIGGER AS
$BODY$
BEGIN
-- TG_TABLE_NAME :name of the table that caused the trigger invocation
END IF ;
RETURN NULL;
END IF ;
END;
$BODY$
LANGUAGEplpgsql VOLATILE
COST 100 ;
DDL_COMM
AND_END
SQL_DROP
This is example for creating an Event Trigger and logging DDL_COMMAND_START events.
CREATE TABLE TAB_EVENT_LOGS (
DATE_TIME TIMESTAMP,
EVENT_NAME TEXT,
REMARKS TEXT
);
The problem with that is that queries typed into the psql console get saved in a history file
.psql_history in the user's home directory and may as well be logged to the PostgreSQL database
server log, thus exposing the password.
To avoid this, use the \ PASSWORD command to set the user password. If the user issuing the
command is a superuser, the current password will not be asked. (Must be superuser to alter
passwords of superusers)
CREATE ROLE niceusername WITH LOGIN ;
\ PASSWORDniceusername
With the above queries, untrusted users can no longer connect to the database.
-- ACCESS SCHEMA
REVOKE ALL ON SCHEMApublic FROMPUBLIC;
GRANT USAGE ON SCHEMApublic TO USER;
The next set of queries revoke all privileges from unauthenticated users and provide limited set
of privileges for the read_write user.
--ACCESS TABLES
REVOKE ALL ON ALL TABLES IN SCHEMA public FROM PUBLIC ;
GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only ;
GRANT SELECT, INSERT, UPDATE, DELETE ON ALL TABLES IN SCHEMA public TO read_write ;
GRANT ALL ON ALL TABLES IN SCHEMA public TO ADMIN ;
--ACCESS SEQUENCES
REVOKE ALL ON ALL SEQUENCES IN SCHEMA public FROM PUBLIC;
GRANT SELECT ON ALL SEQUENCES IN SCHEMA public TO read_only; -- allows the use of CURRVAL
GRANT UPDATE ON ALL SEQUENCES IN SCHEMA public TO read_write; -- allows the use of NEXTVAL and
SETVAL
GRANT USAGE ON ALL SEQUENCES IN SCHEMA public TO read_write; -- allows the use of CURRVAL and
NEXTVAL
GRANT ALL ON ALL SEQUENCES IN SCHEMA public TO ADMIN;
This assumes that pg_hba.conf has been properly configured, which probably looks like this:
2. Set search_path with ALTER USER command to append a new schema my_schema
Alternative:
postgres = # SET ROLE user1;
postgres = # SHOWsearch_path;
search_path
-------------
my_schema, "$user" , public
( 1 ROW)
With below queries, you can set access privileges on objects created in the future in specified
schema.
ALTER DEFAULT PRIVILEGES IN SCHEMA myschema GRANT SELECT ON TABLES TO read_only;
ALTER DEFAULT PRIVILEGES IN SCHEMA myschema GRANT SELECT,INSERT,DELETE,UPDATE ON TABLES TO
read_write; ALTER DEFAULT PRIVILEGES IN SCHEMA myschema GRANT ALL ON TABLES TO
ADMIN; Or, you can set access privileges on objects created in the future by specified user.
ALTER DEFAULT PRIVILEGES FOR ROLE ADMIN GRANT SELECT ON TABLES TO read_only;
Examples:
Only a single comment(string) can be given on any database object. COMMENT will help us to
know what for the particular database object has been defined whats its actual purpose is.
The rule for COMMENT ON ROLE is that you must be superuser to comment on a superuser role,
or have the
CREATEROLE privilege to comment on non-superuser roles. Of course, a superuser can comment
The -Fc selects the "custom backup format" which gives you more power than raw SQL; see
pg_restore for more details. If you want a vanilla SQL file, you can do this instead:
or even
pg_dump DATABASE > DATABASE.sql
A safer alternative uses -1 to wrap the restore in a transaction. The -f specifies the filename
rather than using shell redirection.
psql -1f backup.sql
Custom format files must be restored using pg_restore with the -d option to specify the database:
pg_restore -d DATABASE DATABASE.pgsql
Usage of the custom format is recommended because you can choose which things to restore
and optionally enable parallel processing.
You may need to do a pg_dump followed by a pg_restore if you upgrade from one postgresql
This works behind the scenes by making multiple connections to the server once for each
database and executing pg_dump on it.
Sometimes, you might be tempted to set this up as a cron job, so you want to see the date the
backup was taken as part of the filename:
$ postgres-backup-$ ( DATE +%Y-%m-%d ) .sql
However, please note that this could produce large files on a daily basis. Postgresql has a much
better mechanism for regular backups - WAL archives
The output from pg_dumpall is sufficient to restore to an identically-configured Postgres
instance, but the configuration files in $PGDATA (pg_hba.conf and postgresql.conf) are not part of the
backup, so you'll have to back them up separately.
postgres = # SELECT pg_start_backup ( 'my-backup' );
postgres = # SELECT pg_stop_backup ();
To take a filesystem backup, you must use these functions to help ensure that Postgres is in a
consistent state while the backup is prepared. Section 23.4: Using psql to
export data
Data can be exported using copy command or by taking use of command line options of psql
command.
psql -p \<port> -U \<username> -d \<DATABASE> -A -F<DELIMITER> -c\<sql TO EXECUTE> \> \<output filename
WITH path> psql -p 5432 -U postgres -d test_database -A -F, -c "select * from user" >
/home/USER/user_data.CSV
-F is to specify delimiter
- A OR -- no-align
To insert into table USER from a file named user_data.CSV placed inside /home/USER
/:
Note: In absence of the option WITH DELIMITER , the default delimiter is comma ,
Header option:
TO '/home/user/user_data';
COPY (sql STATEMENT) TO '<filename with path>'; COPY (SELECT * FROM USER WHERE user_name LIKE 'A%')
To Copy into a compressed file
COPY USER TO PROGRAM 'gzip > /home/user/user_data.gz';
#!/bin/sh
cd / save_db
#rm -R /save_db/*
DATE= $ ( date + %d- %m- %Y- %Hh%M)
echo - e "Sauvegarde de la base du ${DATE} "
mkdir prodDir ${DATE}
cd prodDir ${DATE}
#dump file
/ opt / postgres / 9.0 / bin / pg_dump - i - h localhost - p 5432 - U postgres -Fc -b -w -v -f
"dbprod ${DATE} .backup" dbprod
#SQL file
/ opt / postgres / 9.0 / bin / pg_dump - i - h localhost - p 5432 - U postgres -- format plain -- verbose -f
"dbprod ${DATE} .sql" dbprod
During compilation, you have to add the PostgreSQL include directory, which can be pg_config --
found with includedir, to the include path.
You must link with the PostgreSQL client shared library ( libpq.so on UNIX, libpq.dll on Windows).
This library is in the PostgreSQL library directory, which can pg_config --libdir be found with .
Note: For historical reason, the library is called libpq.soand not libpg.so, which is a popular trap for
beginners.
Given that the below code sample is in file coltype.c, compilation and linking would be done with
with the GNU C compiler (consider adding -Wl,-rpath,"$(pg_config --libdir)" to add the
library search path) or
with
cl /MT /W4 /I < include directory > coltype.c < path TO libpq.lib >
Sample program
/* necessary for all PostgreSQL client programs, should be first */
#include <libpq-fe.h>
#include <stdio.h>
#include <string.h>
#ifdef TRACE
#define TRACEFILE "trace.out"
#endif
/*
* Using an empty connectstring will use default values for everything.
* If set, the environment variables PGHOST, PGDATABASE, PGPORT and
* PGUSER will be used.
*/
conn = PQconnectdb ( "" ) ;
/*
* This can only happen if there is not enough memory * to allocate the
PGconn structure. */ if (conn == NULL)
{ fprintf(stderr, "Out of memory connecting to PostgreSQL.\n"); return 1; }
db_host = 'postgres.server.com'
db_port = '5432'
db_un = 'user'
db_pw = 'password'
db_name = 'testdb'
Will result:
[ { 'id' : 2, 'fruit' : 'apple' }, { 'id' : 3, 'fruit' : 'orange' }]
A typical query is performed by creating a command, binding parameters, and then executing
the command. In C#:
var connString = "Host=myserv;Username=myuser;Password=mypass;Database=mydb" ;
using ( var conn = new NpgsqlConnection ( connString ) )
{
var querystring = "INSERT INTO data (some_field) VALUES (@content)" ;
conn . Open ( ) ;
// Create a new command with CommandText and Connection constructor
using ( var cmd = new NpgsqlCommand ( querystring, conn ))
{
// Add a parameter and set its type with the NpgsqlDbType enum
var contentString = "Hello World!" ;
cmd . Parameters . Add( "@content" , NpgsqlDbType . Text ) . Value = contentString ;
/* It is possible to reuse a command object and open connection instead of creating new ones
*/
// Execute the command and read through the rows one by one
using ( NpgsqlDataReader reader = cmd . ExecuteReader ( ) )
{
while ( reader . Read ( ) ) // Returns false for 0 rows, or after reading the last row of
the results
{
// read an integer value
int primaryKey = reader . GetInt32 ( 0 ) ;
// or
primaryKey = Convert . ToInt32 ( reader [ "primary_key" ] );
Assuming, Pomm has been installed using composer, here is a complete example:
< ?php
use PommProject\Foundation\Pomm ;
$loader = require __DIR__ . '/vendor/autoload.php' ;
$pomm = new Pomm( [ 'my_db' => [ 'dsn' => 'pgsql://user:pass@host:5432/db_name' ] ] );
// TABLE comment (
// comment_id uuid PK, created_at timestamptz NN,
// is_moderated bool NN default false,
// content text NN CHECK (content !~ '^\s+$'), author_email text NN)
$sql = <<< SQL
SELECT
comment_id,
created_at,
is_moderated,
content,
author_email
FROM comment
INNER JOIN author USING (author_email)
WHERE
age(now(), created_at) < $*::interval
ORDER BY created_at ASC
SQL;
Pomm’s query manager module escapes query arguments to prevent SQL injection. When the
arguments are cast, it also converts them from a PHP representation to valid Postgres values.
The result is an iterator, it uses a cursor internally. Every row is converted on the fly, booleans to
booleans, timestamps to \DateTime etc.
Chapter 26: Connect to PostgreSQL from
Java
The API to use a relational database from Java is JDBC.
To use it, put the JAR-file with the driver on the JAVA class path.
This documentation shows samples how to use the JDBC driver to connect to a
First, the driver has to be registered with java.sql.DriverManager so that it knows which class to use.
This is done by loading the driver class, typically with java.lang.CLASS.forname( ;driver class
name>).
/**
* Connect to a PostgreSQL database.
* @param url the JDBC URL to connect to; must start with "jdbc:postgresql:"
* @param user the username for the connection
* @param password the password for the connection
* @return a connection object for the established connection
* @throws ClassNotFoundException if the driver class cannot be found on the Java class path
* @throws java.sql.SQLException if the connection to the database fails
*/
private static java. sql . Connection connect ( String url, String user, String password )
throws ClassNotFoundException , java. sql . SQLException
{
/*
* Register the PostgreSQL JDBC driver.
* This may throw a ClassNotFoundException.
*/
Class . forName ( "org.postgresql.Driver" );
/*
* Tell the driver manager to connect to the database specified with the URL.
* This may throw an SQLException.
*/
return java. sql . DriverManager . getConnection ( url, user, password );
}
Not that user and password can also be included in the JDBC URL, in which case you don't have
to specify them in the getConnection method call.
Section 26.2: Connecting with java.sql.DriverManager and
Properties
Instead of specifying connection parameters like user and password (see a complete list here) in
the URL or a separate parameters, you can pack them into a java.util.Properties object:
/**
* Connect to a PostgreSQL database.
* @param url the JDBC URL to connect to. Must start with "jdbc:postgresql:"
* @param user the username for the connection
* @param password the password for the connection
* @return a connection object for the established connection
* @throws ClassNotFoundException if the driver class cannot be found on the Java class path
* @throws java.sql.SQLException if the connection to the database fails
*/
private static java. sql . Connection connect ( String url, String user, String password )
throws ClassNotFoundException , java. sql . SQLException
{
/*
* Register the PostgreSQL JDBC driver.
* This may throw a ClassNotFoundException.
*/
Class . forName ( "org.postgresql.Driver" );
java. util . Properties props = new java. util . Properties ( ) ;
props. setProperty ( "user" , user ) ;
props. setProperty ( "password" , password ) ;
/* don't use server prepared statements */
props. setProperty ( "prepareThreshold" , "0" ) ;
/*
* Tell the driver manager to connect to the database specified with the URL.
* This may throw an SQLException.
*/
return java. sql . DriverManager . getConnection ( url, props );
}
return ds ;
}
Once you have created a data source by calling this function, you would use it like this:
/* get a connection from the connection pool */
java.sql. CONNECTIONconn = ds.getConnection ();
/* do some work */
Requirements:
mkdir $PGDATA/archive
This is host base authentication file, contains the setting for client autherntication.
Add below entry:
`hot_standby` logs what IS required TO accept READ ONLY queries ON slave SERVER.
archive_mode = ON
This parameters allows to send WAL segments to archive location using
archive_command parameter.
archive_command = 'test ! -f /path/to/archivedir/%f && cp %p /path/to/archivedir/%f'
Basically what above archive_command does is it copies the WAL segments to archive
directory.
pg_basebackup utility copies the data from primary server data directory to slave data
directory.
$ pg_basebackup -h < PRIMARY IP > -D /var/lib/postgresql/ < VERSION > /main -U replication -v -P --
xlog-method=stream
-h: Specifies the SYSTEM WHERE TO look FOR the PRIMARY SERVER
-xlog-method = stream: This will FORCE the pg_basebackup TO open another CONNECTION AND stream
enough xlog WHILE backup IS running.
It ALSO ensures that fresh backup can be started WITHOUTfailing back TO
USING an archive.
To configure the standby server, you'll edit postgresql.conf and create a new configuration
file named recovery.conf.
hot_standby = ON
This specifies whether you are allowed to run queries while recovering
standby_mode = ON
Set the connection string to the primary server. Replace with the external IP
address of the primary server. Replace with the password for the user named
replication `primary_conninfo = 'host= port=5432 user=replication password='
(Optional) Set the trigger file location:
trigger_file = '/tmp/postgresql.trigger.5432'
The trigger_file path that you specify is the location where you can add a file when
you want the system to fail over to the standby server. The presence of the file
"triggers" the failover. Alternatively, you can use the pg_ctl promote command to
trigger failover.
Start the standby server
You now have everything in place and are ready to bring up the standby server
Attribution
This article is substantially derived from and attributed to How to Set Up PostgreSQL for High
Availability and Replication with Hot Standby, with minor changes in formatting and examples
and some text deleted. The source was published under the Creative Commons Public License
3.0, which is maintained here.
Chapter 28: EXTENSION dblink and
postgres_fdw
Section 28.1: Extention FDW
FDW is an implimentation of dblink it is more helpful, so to use it:
1. Create an extention:
CREATE EXTENSION postgres_fdw;
2. Create SERVER:
CREATE SERVER name_srv FOREIGN DATA WRAPPER postgres_fdw OPTIONS (host 'hostname',
dbname 'bd_name', port '5432');
CREATE USER MAPPING FOR postgres SERVER name_srv OPTIONS(USER 'postgres', PASSWORD 'password');
1. Create EXTENSION :
2. Create SERVER :
SELECT * FROMschema_name.table_name;
For exemple Select some attribute from another table in another database:
Thank You!