mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Fix ruleutils.c's dumping of whole-row Vars in ROW() and VALUES() contexts.
Normally ruleutils prints a whole-row Var as "foo.*". We already knew that that doesn't work at top level of a SELECT list, because the parser would treat the "*" as a directive to expand the reference into separate columns, not a whole-row Var. However, Joshua Yanovski points out in bug #13776 that the same thing happens at top level of a ROW() construct; and some nosing around in the parser shows that the same is true in VALUES(). Hence, apply the same workaround already devised for the SELECT-list case, namely to add a forced cast to the appropriate rowtype in these cases. (The alternative of just printing "foo" was rejected because it is difficult to avoid ambiguity against plain columns named "foo".) Back-patch to all supported branches.
This commit is contained in:
		| @@ -380,6 +380,8 @@ static void appendContextKeyword(deparse_context *context, const char *str, | ||||
| static void removeStringInfoSpaces(StringInfo str); | ||||
| static void get_rule_expr(Node *node, deparse_context *context, | ||||
| 			  bool showimplicit); | ||||
| static void get_rule_expr_toplevel(Node *node, deparse_context *context, | ||||
| 					   bool showimplicit); | ||||
| static void get_oper_expr(OpExpr *expr, deparse_context *context); | ||||
| static void get_func_expr(FuncExpr *expr, deparse_context *context, | ||||
| 			  bool showimplicit); | ||||
| @@ -4297,9 +4299,9 @@ get_values_def(List *values_lists, deparse_context *context) | ||||
|  | ||||
| 			/* | ||||
| 			 * Strip any top-level nodes representing indirection assignments, | ||||
| 			 * then print the result. | ||||
| 			 * then print the result.  Whole-row Vars need special treatment. | ||||
| 			 */ | ||||
| 			get_rule_expr(processIndirection(col, context, false), | ||||
| 			get_rule_expr_toplevel(processIndirection(col, context, false), | ||||
| 								   context, false); | ||||
| 		} | ||||
| 		appendStringInfoChar(buf, ')'); | ||||
| @@ -4691,7 +4693,8 @@ get_target_list(List *targetList, deparse_context *context, | ||||
| 		 * the top level of a SELECT list it's not right (the parser will | ||||
| 		 * expand that notation into multiple columns, yielding behavior | ||||
| 		 * different from a whole-row Var).  We need to call get_variable | ||||
| 		 * directly so that we can tell it to do the right thing. | ||||
| 		 * directly so that we can tell it to do the right thing, and so that | ||||
| 		 * we can get the attribute name which is the default AS label. | ||||
| 		 */ | ||||
| 		if (tle->expr && IsA(tle->expr, Var)) | ||||
| 		{ | ||||
| @@ -7180,7 +7183,8 @@ get_rule_expr(Node *node, deparse_context *context, | ||||
| 						!tupdesc->attrs[i]->attisdropped) | ||||
| 					{ | ||||
| 						appendStringInfoString(buf, sep); | ||||
| 						get_rule_expr(e, context, true); | ||||
| 						/* Whole-row Vars need special treatment here */ | ||||
| 						get_rule_expr_toplevel(e, context, true); | ||||
| 						sep = ", "; | ||||
| 					} | ||||
| 					i++; | ||||
| @@ -7560,6 +7564,27 @@ get_rule_expr(Node *node, deparse_context *context, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * get_rule_expr_toplevel		- Parse back a toplevel expression | ||||
|  * | ||||
|  * Same as get_rule_expr(), except that if the expr is just a Var, we pass | ||||
|  * istoplevel = true not false to get_variable().  This causes whole-row Vars | ||||
|  * to get printed with decoration that will prevent expansion of "*". | ||||
|  * We need to use this in contexts such as ROW() and VALUES(), where the | ||||
|  * parser would expand "foo.*" appearing at top level.  (In principle we'd | ||||
|  * use this in get_target_list() too, but that has additional worries about | ||||
|  * whether to print AS, so it needs to invoke get_variable() directly anyway.) | ||||
|  */ | ||||
| static void | ||||
| get_rule_expr_toplevel(Node *node, deparse_context *context, | ||||
| 					   bool showimplicit) | ||||
| { | ||||
| 	if (node && IsA(node, Var)) | ||||
| 		(void) get_variable((Var *) node, 0, true, context); | ||||
| 	else | ||||
| 		get_rule_expr(node, context, showimplicit); | ||||
| } | ||||
|  | ||||
|  | ||||
| /* | ||||
|  * get_oper_expr			- Parse back an OpExpr node | ||||
|   | ||||
| @@ -1384,6 +1384,97 @@ select * from tt14v; | ||||
|  foo |    | quux | ||||
| (1 row) | ||||
|  | ||||
| -- check display of whole-row variables in some corner cases | ||||
| create type nestedcomposite as (x int8_tbl); | ||||
| create view tt15v as select row(i)::nestedcomposite from int8_tbl i; | ||||
| select * from tt15v; | ||||
|                    row                     | ||||
| ------------------------------------------ | ||||
|  ("(123,456)") | ||||
|  ("(123,4567890123456789)") | ||||
|  ("(4567890123456789,123)") | ||||
|  ("(4567890123456789,4567890123456789)") | ||||
|  ("(4567890123456789,-4567890123456789)") | ||||
| (5 rows) | ||||
|  | ||||
| select pg_get_viewdef('tt15v', true); | ||||
|                     pg_get_viewdef                     | ||||
| ------------------------------------------------------ | ||||
|   SELECT ROW(i.*::int8_tbl)::nestedcomposite AS "row"+ | ||||
|     FROM int8_tbl i; | ||||
| (1 row) | ||||
|  | ||||
| select row(i.*::int8_tbl)::nestedcomposite from int8_tbl i; | ||||
|                    row                     | ||||
| ------------------------------------------ | ||||
|  ("(123,456)") | ||||
|  ("(123,4567890123456789)") | ||||
|  ("(4567890123456789,123)") | ||||
|  ("(4567890123456789,4567890123456789)") | ||||
|  ("(4567890123456789,-4567890123456789)") | ||||
| (5 rows) | ||||
|  | ||||
| create view tt16v as select * from int8_tbl i, lateral(values(i)) ss; | ||||
| select * from tt16v; | ||||
|         q1        |        q2         |               column1                 | ||||
| ------------------+-------------------+-------------------------------------- | ||||
|               123 |               456 | (123,456) | ||||
|               123 |  4567890123456789 | (123,4567890123456789) | ||||
|  4567890123456789 |               123 | (4567890123456789,123) | ||||
|  4567890123456789 |  4567890123456789 | (4567890123456789,4567890123456789) | ||||
|  4567890123456789 | -4567890123456789 | (4567890123456789,-4567890123456789) | ||||
| (5 rows) | ||||
|  | ||||
| select pg_get_viewdef('tt16v', true); | ||||
|               pg_get_viewdef                | ||||
| ------------------------------------------- | ||||
|   SELECT i.q1,                            + | ||||
|      i.q2,                                + | ||||
|      ss.column1                           + | ||||
|     FROM int8_tbl i,                      + | ||||
|      LATERAL ( VALUES (i.*::int8_tbl)) ss; | ||||
| (1 row) | ||||
|  | ||||
| select * from int8_tbl i, lateral(values(i.*::int8_tbl)) ss; | ||||
|         q1        |        q2         |               column1                 | ||||
| ------------------+-------------------+-------------------------------------- | ||||
|               123 |               456 | (123,456) | ||||
|               123 |  4567890123456789 | (123,4567890123456789) | ||||
|  4567890123456789 |               123 | (4567890123456789,123) | ||||
|  4567890123456789 |  4567890123456789 | (4567890123456789,4567890123456789) | ||||
|  4567890123456789 | -4567890123456789 | (4567890123456789,-4567890123456789) | ||||
| (5 rows) | ||||
|  | ||||
| create view tt17v as select * from int8_tbl i where i in (values(i)); | ||||
| select * from tt17v; | ||||
|         q1        |        q2          | ||||
| ------------------+------------------- | ||||
|               123 |               456 | ||||
|               123 |  4567890123456789 | ||||
|  4567890123456789 |               123 | ||||
|  4567890123456789 |  4567890123456789 | ||||
|  4567890123456789 | -4567890123456789 | ||||
| (5 rows) | ||||
|  | ||||
| select pg_get_viewdef('tt17v', true); | ||||
|                pg_get_viewdef                 | ||||
| --------------------------------------------- | ||||
|   SELECT i.q1,                              + | ||||
|      i.q2                                   + | ||||
|     FROM int8_tbl i                         + | ||||
|    WHERE (i.* IN ( VALUES (i.*::int8_tbl))); | ||||
| (1 row) | ||||
|  | ||||
| select * from int8_tbl i where i.* in (values(i.*::int8_tbl)); | ||||
|         q1        |        q2          | ||||
| ------------------+------------------- | ||||
|               123 |               456 | ||||
|               123 |  4567890123456789 | ||||
|  4567890123456789 |               123 | ||||
|  4567890123456789 |  4567890123456789 | ||||
|  4567890123456789 | -4567890123456789 | ||||
| (5 rows) | ||||
|  | ||||
| -- clean up all the random objects we made above | ||||
| set client_min_messages = warning; | ||||
| DROP SCHEMA temp_view_test CASCADE; | ||||
|   | ||||
| @@ -469,6 +469,24 @@ alter table tt14t drop column f3; | ||||
| select pg_get_viewdef('tt14v', true); | ||||
| select * from tt14v; | ||||
|  | ||||
| -- check display of whole-row variables in some corner cases | ||||
|  | ||||
| create type nestedcomposite as (x int8_tbl); | ||||
| create view tt15v as select row(i)::nestedcomposite from int8_tbl i; | ||||
| select * from tt15v; | ||||
| select pg_get_viewdef('tt15v', true); | ||||
| select row(i.*::int8_tbl)::nestedcomposite from int8_tbl i; | ||||
|  | ||||
| create view tt16v as select * from int8_tbl i, lateral(values(i)) ss; | ||||
| select * from tt16v; | ||||
| select pg_get_viewdef('tt16v', true); | ||||
| select * from int8_tbl i, lateral(values(i.*::int8_tbl)) ss; | ||||
|  | ||||
| create view tt17v as select * from int8_tbl i where i in (values(i)); | ||||
| select * from tt17v; | ||||
| select pg_get_viewdef('tt17v', true); | ||||
| select * from int8_tbl i where i.* in (values(i.*::int8_tbl)); | ||||
|  | ||||
| -- clean up all the random objects we made above | ||||
| set client_min_messages = warning; | ||||
| DROP SCHEMA temp_view_test CASCADE; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user