com.hazelcast.jet.sql.impl.opt.logical.UpdateLogicalRules Maven / Gradle / Ivy
/*
* Copyright 2021 Hazelcast Inc.
*
* Licensed under the Hazelcast Community License (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://hazelcast.com/hazelcast-community-license
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.hazelcast.jet.sql.impl.opt.logical;
import com.hazelcast.jet.sql.impl.connector.SqlConnectorUtil;
import com.hazelcast.jet.sql.impl.opt.OptUtils;
import com.hazelcast.jet.sql.impl.schema.HazelcastRelOptTable;
import com.hazelcast.jet.sql.impl.schema.HazelcastTable;
import com.hazelcast.sql.impl.schema.Table;
import com.hazelcast.org.apache.calcite.plan.RelOptRule;
import com.hazelcast.org.apache.calcite.plan.RelOptRuleCall;
import com.hazelcast.org.apache.calcite.rel.RelNode;
import com.hazelcast.org.apache.calcite.rel.core.TableModify;
import com.hazelcast.org.apache.calcite.rel.logical.LogicalProject;
import com.hazelcast.org.apache.calcite.rel.logical.LogicalTableModify;
import com.hazelcast.org.apache.calcite.rel.logical.LogicalTableScan;
import com.hazelcast.org.apache.calcite.rel.logical.LogicalValues;
import java.util.List;
import java.util.stream.IntStream;
import static java.util.stream.Collectors.toList;
import static com.hazelcast.org.apache.calcite.plan.RelOptRule.none;
import static com.hazelcast.org.apache.calcite.plan.RelOptRule.operand;
import static com.hazelcast.org.apache.calcite.plan.RelOptRule.operandJ;
// no support for UPDATE FROM SELECT case (which is not yet implemented)
// once joins are there we need to create complementary rule
final class UpdateLogicalRules {
@SuppressWarnings("checkstyle:anoninnerlength")
static final RelOptRule INSTANCE =
new RelOptRule(
operandJ(
LogicalTableModify.class, null, TableModify::isUpdate,
operand(
LogicalProject.class,
operand(LogicalTableScan.class, none())
)
),
UpdateLogicalRules.class.getSimpleName()
) {
@Override
public void onMatch(RelOptRuleCall call) {
LogicalTableModify update = call.rel(0);
LogicalTableScan scan = call.rel(2);
UpdateLogicalRel rel = new UpdateLogicalRel(
update.getCluster(),
OptUtils.toLogicalConvention(update.getTraitSet()),
update.getTable(),
update.getCatalogReader(),
rewriteScan(scan),
update.getUpdateColumnList(),
update.getSourceExpressionList(),
update.isFlattened()
);
call.transformTo(rel);
}
// rewrites existing project to just primary keys project
private RelNode rewriteScan(LogicalTableScan scan) {
HazelcastRelOptTable relTable = (HazelcastRelOptTable) scan.getTable();
HazelcastTable hazelcastTable = relTable.unwrap(HazelcastTable.class);
return new FullScanLogicalRel(
scan.getCluster(),
OptUtils.toLogicalConvention(scan.getTraitSet()),
OptUtils.createRelTable(
relTable.getDelegate().getQualifiedName(),
hazelcastTable.withProject(keyProjects(hazelcastTable.getTarget())),
scan.getCluster().getTypeFactory()
)
);
}
private List keyProjects(Table table) {
List primaryKey = SqlConnectorUtil.getJetSqlConnector(table).getPrimaryKey(table);
return IntStream.range(0, table.getFieldCount())
.filter(i -> primaryKey.contains(table.getField(i).getName()))
.boxed()
.collect(toList());
}
};
// no-updates case, i.e. '... WHERE __key = 1 AND __key = 2'
// could/should be optimized to no-op
static final RelOptRule NOOP_INSTANCE =
new RelOptRule(
operandJ(
LogicalTableModify.class, null, TableModify::isUpdate,
operand(LogicalValues.class, none())
),
UpdateLogicalRules.class.getSimpleName() + "(NOOP)"
) {
@Override
public void onMatch(RelOptRuleCall call) {
LogicalTableModify update = call.rel(0);
LogicalValues values = call.rel(1);
UpdateLogicalRel rel = new UpdateLogicalRel(
update.getCluster(),
OptUtils.toLogicalConvention(update.getTraitSet()),
update.getTable(),
update.getCatalogReader(),
OptUtils.toLogicalInput(values),
update.getUpdateColumnList(),
update.getSourceExpressionList(),
update.isFlattened()
);
call.transformTo(rel);
}
};
private UpdateLogicalRules() {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy