É possível ao usar o pgadmin
ou plsql
para obter um porão de um plano de consulta para uma instrução SQL executado dentro de um u Ser d efined f unção (UDF) utilizando EXPLAIN
. Então, como eu me aposso do plano de consulta para uma chamada específica de uma UDF? Vejo o UDF abstraído em uma única operação F()
no pgadmin.
Examinei a documentação, mas não consegui encontrar nada.
Atualmente, estou retirando as instruções e executando-as manualmente. Mas isso não serve para consultas grandes.
Por exemplo, considere o UDF abaixo. Esse UDF, mesmo que tenha a capacidade de imprimir sua sequência de consultas, não funcionará com uma cópia-colar, pois possui uma tabela temporária criada local, que não existe quando você a cola e executa.
CREATE OR REPLACE FUNCTION get_paginated_search_results(
forum_id_ INTEGER,
query_ CHARACTER VARYING,
from_date_ TIMESTAMP WITHOUT TIME ZONE DEFAULT NULL,
to_date_ TIMESTAMP WITHOUT TIME ZONE DEFAULT NULL,
in_categories_ INTEGER[] DEFAULT '{}')
RETURNS SETOF post_result_entry AS $$
DECLARE
join_string CHARACTER VARYING := ' ';
from_where_date CHARACTER VARYING := ' ';
to_where_date CHARACTER VARYING := ' ';
query_string_ CHARACTER VARYING := ' ';
BEGIN
IF NOT from_date_ IS NULL THEN
from_where_date := ' AND fp.posted_at > ''' || from_date_ || '''';
END IF;
IF NOT to_date_ IS NULL THEN
to_where_date := ' AND fp.posted_at < ''' || to_date_ || '''';
END IF;
CREATE LOCAL TEMP TABLE un_cat(id) ON COMMIT DROP AS (select * from unnest(in_categories_)) ;
if in_categories_ != '{}' THEN
join_string := ' INNER JOIN forum_topics ft ON fp.topic_id = ft.id ' ||
' INNER JOIN un_cat uc ON uc.id = ft.category_id ' ;
END IF;
query_string_ := '
SELECT index,posted_at,post_text,name,join_date,quotes
FROM forum_posts fp
INNER JOIN forum_user fu ON
fu.forum_id = fp.forum_id AND fu.id = fp.user_id' ||
join_string
||
'WHERE fu.forum_id = ' || forum_id_ || ' AND
to_tsvector(''english'',fp.post_text) @@ to_tsquery(''english'','''|| query_||''')' ||
from_where_date ||
to_where_date
||';';
RAISE NOTICE '%', query_string_ ;
RETURN QUERY
EXECUTE query_string_;
END;
$$ LANGUAGE plpgsql;
fonte
Além do conselho da @ rfusca: as instruções SQL dentro das funções plpgsql são consideradas instruções aninhadas e você precisa definir o Parâmetro adicional
auto_explain.log_nested_statements
.Ao contrário de outras extensões, você não precisa executar
CREATE EXTENSION
para esta. Basta carregá-lo dinamicamente em sua sessão comLOAD
. Sua sessão pode ficar assim:Pode produzir muita saída de log.
O manual atual sobre auto_explain.
Depesz escreveu um artigo no blog quando foi introduzido no PostgreSQL 8.4.
fonte
LOAD 'auto_explain';
retornaERROR: access to library "auto_explain" is not allowed
. O que nesse caso? Eu tive algum sucesso cortando minhas funções,return query explain select …
mas isso é trabalhoso e lento.