-
Notifications
You must be signed in to change notification settings - Fork 6.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fix on conflict
error and add support for other on conflict
queries in Postgres
#34227
base: master
Are you sure you want to change the base?
Fix on conflict
error and add support for other on conflict
queries in Postgres
#34227
Conversation
…ement of postgres
…or_postgres' into Issue#32280_bugfix_on_conflict_for_postgres
on conflict
error and add support for other on conflict
queries in Postgres
Hi @terrymanu, Will you please review and proceed? |
@omkar-shitole Thank you for your contribution, I will review this pr later. |
RELEASE-NOTES.md
Outdated
@@ -75,6 +75,7 @@ | |||
1. Encrypt: Use sql bind info in EncryptInsertPredicateColumnTokenGenerator to avoid wrong column table mapping - [#34110](https://github.com/apache/shardingsphere/pull/34110) | |||
1. Mode: Fixes `JDBCRepository` improper handling of H2-database in memory mode - [#33281](https://github.com/apache/shardingsphere/issues/33281) | |||
1. Mode: Fixes duplicate column names added when index changed in DDL - [#33982](https://github.com/apache/shardingsphere/issues/33281) | |||
1. SQL Binder: Fixes bug: throwing exception while using WHERE statement in ON CONFLICT with INSERT INTO in Postgres [#32280](https://github.com/apache/shardingsphere/issues/32280) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you modify this change log to Fix on conflict error and add support for other on conflict queries in Postgres #34227
?
import java.util.List; | ||
import java.util.Optional; | ||
import java.util.stream.Collectors; | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please add javadoc for this class.
import java.util.stream.Collectors; | ||
|
||
@Getter | ||
@ToString |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why add @ToString
annotation?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for pointing this out. I added the annotation because, while it has not been explicitly used anywhere currently, I observed a similar annotation unused applied to the OnDuplicateUpdateContext
class. I included it to maintain consistency and to ensure that if needed later.
However, if you feel this annotation is unnecessary for now, let me know will cover this in the correction commit.
|
||
@Getter | ||
@ToString | ||
public final class OnConflictUpdateContext implements WhereAvailable { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please remove WhereAvailable here, it's only for SQLStatement.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for reviewing and pointing this out too. Adding it here was redundant and an oversight on my part as I missed removing it during cleanup. I will address it in the correction commit.
Thank you again!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@strongduanmu
All other suggested changes/modifications are ready and added to the branch.
Will you please confirm this once?
|
||
private final Collection<ColumnSegment> columnSegments; | ||
|
||
private final Collection<BinaryOperationExpression> joinConditions = new LinkedList<>(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is there a join condition in On Conflict
segment?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No, It's not part of a particular segment. join can be handled outside the conflict segment in Insert statement . I'll address this in a correction commit.
https://www.postgresql.org/docs/current/sql-insert.html
…ConflictupdateContext
public OnConflictUpdateContext(final Collection<ColumnAssignmentSegment> assignments, final List<Object> params, final int parametersOffset, final Optional<WhereSegment> segment) { | ||
List<ExpressionSegment> expressionSegments = assignments.stream().map(ColumnAssignmentSegment::getValue).collect(Collectors.toList()); | ||
segment.ifPresent(whereSegments::add); | ||
for (WhereSegment whereSegment : whereSegments) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please rename whereSegment to each
for (WhereSegment whereSegment : whereSegments) { | ||
expressionSegments.add(whereSegment.getExpr()); | ||
} | ||
columnSegments = assignments.stream().map(each -> each.getColumns().get(0)).collect(Collectors.toList()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why only call each.getColumns().get(0)
here?
parameters = getParameters(params, parametersOffset); | ||
} | ||
|
||
/** |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please remove javadoc for private method.
* @return List of value expressions | ||
*/ | ||
private List<ExpressionSegment> getValueExpressions(final Collection<ExpressionSegment> assignments) { | ||
List<ExpressionSegment> result = new ArrayList<>(assignments.size()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Only for cast Collection to List? If so, please modify the origianl assignments to ArrayList.
return result; | ||
} | ||
|
||
/** |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please remove javadoc for private method.
import static org.hamcrest.CoreMatchers.is; | ||
import static org.hamcrest.MatcherAssert.assertThat; | ||
|
||
public class OnConflictUpdateContextTest { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please remove public modifier for unit test class.
parameterBuilder = containsInsertValues(sqlStatementContext) | ||
? new GroupedParameterBuilder(((InsertStatementContext) sqlStatementContext).getGroupedParameters(), ((InsertStatementContext) sqlStatementContext).getOnDuplicateKeyUpdateParameters()) | ||
: new StandardParameterBuilder(parameters); | ||
if (sqlStatementContext.getSqlStatement() instanceof PostgreSQLInsertStatement) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please don't use PostgreSQLInsertStatement here. This method is not pluggable.
when(queryContext.getSql()).thenReturn("INSERT INTO tbl VALUES (?)"); | ||
when(queryContext.getParameters()).thenReturn(parameters); | ||
when(queryContext.getHintValueContext()).thenReturn(hintValueContext); | ||
return queryContext; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please rename queryContext to result.
@@ -704,7 +704,7 @@ public ASTNode visitInsert(final InsertContext ctx) { | |||
PostgreSQLInsertStatement result = (PostgreSQLInsertStatement) visit(ctx.insertRest()); | |||
result.setTable((SimpleTableSegment) visit(ctx.insertTarget())); | |||
if (null != ctx.optOnConflict()) { | |||
result.setOnDuplicateKeyColumnsSegment((OnDuplicateKeyColumnsSegment) visit(ctx.optOnConflict())); | |||
result.setOnConflictKeyColumnsSegment((OnConflictKeyColumnsSegment) visit(ctx.optOnConflict())); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is the difference between OnDuplicateKeyColumnsSegment and OnConflictKeyColumnsSegment? If they are the same functionality, it may be better to use the same Segment.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please let me know what pattern should be followed in this case.
Postgres has ON CONFLICT
with the WHERE
statement. Also, there is DO NOTHING
inside Postgres in Combination with ON ONCLFICT
.
Mysql has ON DUPLICATE
without a WHERE
segment and apart from that there is no as such difference between them.
@@ -719,7 +719,11 @@ public ASTNode visitOptOnConflict(final OptOnConflictContext ctx) { | |||
if (null != ctx.setClauseList()) { | |||
assignments = ((SetAssignmentSegment) visit(ctx.setClauseList())).getAssignments(); | |||
} | |||
return new OnDuplicateKeyColumnsSegment(ctx.getStart().getStartIndex(), ctx.getStop().getStopIndex(), assignments); | |||
WhereSegment whereSegment = | |||
ctx.whereClause() != null |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please put null to the left side.
Fixes #32280
Changes proposed in this pull request:
whereSegment
inonConflictKeyColumnSegment
which was missing and required. The current flow is more intent on theon_duplicate
of MySQL thanon_conflict
in Postgres which has been added in Postgres after Postgres 9.optOnConflict
rule which was eventually failing to read parameter markers expressions and assignments present inwhereSegment
INSERT INTO
statement. The flow for ON DUPLICATE has not been removed or altered for now, as It may be useful for certain conditions of older versions of Postgres (and can be deprecated in future if no use case is observed for long)Before committing this PR, I'm sure that I have checked the following options:
./mvnw clean install -B -T1C -Dmaven.javadoc.skip -Dmaven.jacoco.skip -e
.