I've this function that inserts a row into a city
table without duplicates. It returns the id of the inserted row:
CREATE OR REPLACE FUNCTION public.insert_city(
character varying,
character varying,
character varying,
character varying,
character varying,
character varying)
RETURNS integer AS
$BODY$
DECLARE
name_city1 ALIAS FOR $1;
country1 ALIAS FOR $2;
province1 ALIAS FOR $3;
region1 ALIAS FOR $4;
cap1 ALIAS FOR $5;
nationality1 ALIAS FOR $6;
id_city1 integer;
BEGIN
INSERT INTO city (name_city, country, province, region, cap, nationality)
SELECT name_city1, country1, province1, region1, cap1, nationality1
WHERE NOT EXISTS (SELECT id_city FROM city WHERE name_city = name_city1)
RETURNING id_city INTO id_city1;
-- xxx
END;
$BODY$
LANGUAGE plpgsql VOLATILE;
xxx
marks the spot where I need something like this:
IF is_number(id_city1) THEN
RETURN id_city1;
ELSE
RETURN query select id_city from city where name_city=name_city1;
END IF;
If the first query does not insert a new row and I don't get an id_city
from it, I want to execute the second query to select an existing id_city
.
How can I do this?
Your function can be simplified some more. More importantly, you can fix the built-in race condition:
Major points
Assuming you run Postgres 9.5 or later, since you did not declare it.
Use the new faster UPSERT solution
INSERT .. ON CONFLICT ...
Detailed explanation:
You need a
UNIQUE
constraint onname_city
for this.About
UNION ALL ... LIMIT 1
:Can be achieved with a single SQL command using a data-modifying CTE. This is least vulnerable to lock contention or other concurrency issues. It's shortest and fastest even without concurrent access.
The function can be a simpler SQL function. (But plpgsql isn't wrong or bad either.)
Don't abuse
to attach names to parameters. That's explicitly discouraged in the manual. Use proper parameter names. The manual:ALIAS FOR
This is plpgsql version
There is a way that when the row exist don't increase the primary_key number (id_city in this case) ?
Why not alter your function like so?:
Insert the existing
id_city
intoid_city1
. If one does not exist, it will beNULL
. You can then perform theINSERT
if it isNULL
and assign the newid_city1
. Finally returnid_city1
.