fill_extraUpdatedCols is done in completely the wrong place

Started by Tom Laneover 5 years ago3 messages
#1Tom Lane
tgl@sss.pgh.pa.us

I happened to notice $subject while working on the release notes.
AFAICS, it is 100% inappropriate for the parser to compute the
set of generated columns affected by an UPDATE, because that set
could change before execution. It would be really easy to break
this for an UPDATE in a stored rule, for example.

I think that that processing should be done by the planner, instead.
I don't object too much to keeping the data in RTEs ... but there had
better be a bold annotation that the set is not valid till after
planning.

An alternative solution is to keep the set in some executor data structure
and compute it during executor startup; perhaps near to where the
expressions are prepared for execution, so as to save extra stringToNode
calls.

regards, tom lane

#2Peter Eisentraut
peter.eisentraut@2ndquadrant.com
In reply to: Tom Lane (#1)
Re: fill_extraUpdatedCols is done in completely the wrong place

On 2020-05-08 21:05, Tom Lane wrote:

I happened to notice $subject while working on the release notes.
AFAICS, it is 100% inappropriate for the parser to compute the
set of generated columns affected by an UPDATE, because that set
could change before execution. It would be really easy to break
this for an UPDATE in a stored rule, for example.

Do you have a specific situation in mind? How would a rule change the
set of columns updated by a query? Something involving CTEs? Having a
test case would be good.

I think that that processing should be done by the planner, instead.
I don't object too much to keeping the data in RTEs ... but there had
better be a bold annotation that the set is not valid till after
planning.

An alternative solution is to keep the set in some executor data structure
and compute it during executor startup; perhaps near to where the
expressions are prepared for execution, so as to save extra stringToNode
calls.

Yeah, really only the executor ended up needing this, so perhaps it
should be handled in the executor.

--
Peter Eisentraut http://www.2ndQuadrant.com/
PostgreSQL Development, 24x7 Support, Remote DBA, Training & Services

#3Tom Lane
tgl@sss.pgh.pa.us
In reply to: Peter Eisentraut (#2)
2 attachment(s)
Re: fill_extraUpdatedCols is done in completely the wrong place

Peter Eisentraut <peter.eisentraut@2ndquadrant.com> writes:

On 2020-05-08 21:05, Tom Lane wrote:

I happened to notice $subject while working on the release notes.
AFAICS, it is 100% inappropriate for the parser to compute the
set of generated columns affected by an UPDATE, because that set
could change before execution. It would be really easy to break
this for an UPDATE in a stored rule, for example.

Do you have a specific situation in mind? How would a rule change the
set of columns updated by a query? Something involving CTEs? Having a
test case would be good.

broken-update-rule.sql, attached, shows the scenario I had in mind:
the rule UPDATE query knows nothing of the generated column that
gets added after the rule is stored, so the UPDATE fails to update it.

However, on the way to preparing that test case I discovered that
auto-updatable views have the same disease even when the generated column
exists from the get-go; see broken-updatable-views.sql. In the context
of the existing design, I suppose this means that there needs to be
a fill_extraUpdatedCols call somewhere in the code path that constructs
an auto-update query. But if we moved the whole thing to the executor
then the problem would go away.

I observe also that the executor doesn't seem to need this bitmap at all
unless (a) there are triggers or (b) there are generated columns.
So in a lot of simpler cases, the cost of doing fill_extraUpdatedCols
at either parse or plan time would be quite wasted. That might be a good
argument for moving it to executor start, even though we'd then have
to re-do it when re-using a prepared plan.

regards, tom lane

Attachments:

broken-update-rule.sqltext/plain; charset=us-ascii; name=broken-update-rule.sqlDownload
broken-updatable-views.sqltext/plain; charset=us-ascii; name=broken-updatable-views.sqlDownload