Last active
June 21, 2024 01:26
-
-
Save insinfo/55f14bc4fcdbf6995371876b8c314e13 to your computer and use it in GitHub Desktop.
create serial by year in postgresql
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
CREATE OR REPLACE FUNCTION serial_year_func() | |
RETURNS "pg_catalog"."trigger" AS $BODY$ | |
DECLARE | |
current_year INT; | |
table_name TEXT; | |
id_field_name TEXT; | |
ano_field_name TEXT; | |
last_id INT; | |
_idIsNull boolean; | |
_anoIsNull boolean; | |
result RECORD; | |
BEGIN | |
-- based in https://stackoverflow.com/questions/7711432/how-to-set-value-of-composite-variable-field-using-dynamic-sql/7782641#7782641 | |
-- https://gist.github.com/insinfo/55f14bc4fcdbf6995371876b8c314e13 | |
-- for debug https://www.postgresonline.com/journal/archives/214-Using-PgAdmin-PLPgSQL-Debugger.html | |
-- CREATE extension pldbgapi; | |
current_year := EXTRACT(YEAR FROM NOW()); | |
table_name := TG_ARGV[0]; | |
id_field_name := TG_ARGV[1]; | |
ano_field_name := TG_ARGV[2]; | |
result := NEW; | |
-- verifica se esta sendo passado o id manualmente no insert statment | |
execute 'select $1.'|| quote_ident(id_field_name) ||' is null' using NEW into _idIsNull; | |
-- verifica se esta sendo passado o ano manualmente no insert statement | |
execute 'select $1.'|| quote_ident(ano_field_name) ||' is null' using NEW into _anoIsNull; | |
IF _idIsNull THEN | |
-- pega o ultimo registro cadastrado na tabela | |
EXECUTE format('SELECT %I FROM %I WHERE %I = $1 ORDER BY %I DESC LIMIT 1', id_field_name, table_name,ano_field_name, id_field_name) | |
INTO last_id | |
USING current_year; | |
IF last_id IS NULL THEN | |
last_id := 0; | |
END IF; | |
-- set NEW.id | |
create temp TABLE IF NOT EXISTS aux as select NEW.*; | |
execute 'update aux set ' || quote_ident(id_field_name) || ' = ' || last_id + 1; | |
select into result * from aux; | |
-- with hstore CREATE EXTENSION hstore; | |
--NEW := NEW #= hstore(id_field_name, (last_id + 1)::text) ; | |
-- with json not work | |
--EXECUTE 'SELECT json_populate_record($1, json_build_object(' || quote_literal(id_field_name) || ', $2))' INTO result USING NEW, last_id + 1; | |
--result2 := result; | |
END IF; | |
IF _anoIsNull THEN | |
-- set NEW.ano_exercicio | |
create temp TABLE IF NOT EXISTS aux as select NEW.*; | |
execute 'update aux set ' || quote_ident(ano_field_name) || ' = ' || current_year; | |
select into result * from aux; | |
--NEW := NEW #= hstore(ano_field_name, current_year::text ); | |
--EXECUTE 'SELECT json_populate_record($1, json_build_object(' || quote_literal(ano_field_name) || ', $2))' INTO result USING NEW, current_year; | |
END IF; | |
DROP TABLE IF EXISTS aux; | |
RETURN result; | |
END; | |
$BODY$ | |
LANGUAGE plpgsql VOLATILE | |
COST 100 |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
The serial_anual_func function is a PostgreSQL trigger function designed to automatically assign a sequential ID and the current year to specified fields in a table when a new record is inserted. This function ensures that the ID and year fields are populated correctly if they are not manually provided during the insert.
Explanation of the Function
Variable Initialization: Initializes variables to store current year, table name, ID field name, year field name, last ID, and boolean flags to check if fields are null.
Check for Null ID:
Executes a query to check if the ID is null in the NEW record.
If null, fetches the last inserted ID for the current year, increments it by 1, and assigns it to the ID field of NEW.
Check for Null Year:
Executes a query to check if the year is null in the NEW record.
If null, assigns the current year to the year field of NEW.
Temporary Table Usage:
Uses a temporary table aux to store and update the NEW record values for both ID and year fields.
Ensures that the updates are correctly applied to the NEW record by selecting the modified record back into result.
Return Updated Record: Returns the modified NEW record.
Usage
To create a trigger using this function, use the following command:
Replace your_table_name, id_field_name, and ano_field_name with the actual table and field names. This trigger will ensure that the id_field_name and ano_field_name are automatically populated if not provided during an insert.