Skip to content

Commit 8874d3e

Browse files
authored
Merge pull request #30666 from ggevay/insert-opt
Create a special optimizer pipeline for constant INSERTs
2 parents 2c0f039 + a37c334 commit 8874d3e

File tree

8 files changed

+61
-14
lines changed

8 files changed

+61
-14
lines changed

misc/python/materialize/feature_benchmark/scenarios/benchmark_main.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -311,7 +311,10 @@ def benchmark(self) -> MeasurementSource:
311311

312312

313313
class InsertMultiRow(DML):
314-
"""Measure the time it takes for a single multi-row INSERT statement to return."""
314+
"""Measure the time it takes for a single multi-row INSERT statement to return.
315+
When `sequence_insert` calls `constant_optimizer`, it should be able to reach a constant. Otherwise, we run the full
316+
logical optimizer, which makes this test show a regression.
317+
"""
315318

316319
SCALE = 4 # FATAL: request larger than 2.0 MB
317320

src/adapter/src/catalog/state.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -1200,13 +1200,12 @@ impl CatalogState {
12001200
}
12011201
Some(_) | None => {
12021202
let optimizer_features = optimizer_config.features.clone();
1203-
// Build an optimizer for this VIEW.
12041203
// TODO(aalexandrov): ideally this should be a materialized_view::Optimizer.
12051204
let mut optimizer = optimize::view::Optimizer::new(optimizer_config, None);
12061205

12071206
let raw_expr = materialized_view.expr;
12081207
let optimized_expr = match optimizer.optimize(raw_expr.clone()) {
1209-
Ok(optimzed_expr) => optimzed_expr,
1208+
Ok(optimized_expr) => optimized_expr,
12101209
Err(err) => return Err((err.into(), cached_expr)),
12111210
};
12121211

src/adapter/src/client.rs

-1
Original file line numberDiff line numberDiff line change
@@ -741,7 +741,6 @@ impl SessionClient {
741741

742742
// Collect optimizer parameters.
743743
let optimizer_config = optimize::OptimizerConfig::from(conn_catalog.system_vars());
744-
// Build an optimizer for this VIEW.
745744
let mut optimizer = optimize::view::Optimizer::new(optimizer_config, None);
746745

747746
let result: Result<_, AdapterError> =

src/adapter/src/coord/sequencer/inner.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2647,7 +2647,7 @@ impl Coordinator {
26472647
// Collect optimizer parameters.
26482648
let optimizer_config = optimize::OptimizerConfig::from(self.catalog().system_config());
26492649

2650-
// Build an optimizer for this VIEW.
2650+
// (`optimize::view::Optimizer` has a special case for constant queries.)
26512651
let mut optimizer = optimize::view::Optimizer::new(optimizer_config, None);
26522652

26532653
// HIR ⇒ MIR lowering and MIR ⇒ MIR optimization (local)

src/adapter/src/coord/sequencer/inner/explain_timestamp.rs

-1
Original file line numberDiff line numberDiff line change
@@ -130,7 +130,6 @@ impl Coordinator {
130130
// Collect optimizer parameters.
131131
let optimizer_config = optimize::OptimizerConfig::from(self.catalog().system_config());
132132

133-
// Build an optimizer for this VIEW.
134133
let mut optimizer = optimize::view::Optimizer::new(optimizer_config, None);
135134

136135
let span = Span::current();

src/adapter/src/optimize.rs

+16
Original file line numberDiff line numberDiff line change
@@ -359,6 +359,22 @@ fn optimize_mir_local(
359359
Ok::<_, OptimizerError>(expr)
360360
}
361361

362+
/// This is just a wrapper around [mz_transform::Optimizer::constant_optimizer],
363+
/// running it, and tracing the result plan.
364+
#[mz_ore::instrument(target = "optimizer", level = "debug", name = "constant")]
365+
fn optimize_mir_constant(
366+
expr: MirRelationExpr,
367+
ctx: &mut TransformCtx,
368+
) -> Result<MirRelationExpr, OptimizerError> {
369+
let optimizer = mz_transform::Optimizer::constant_optimizer(ctx);
370+
let expr = optimizer.optimize(expr, ctx)?;
371+
372+
// Trace the result of this phase.
373+
mz_repr::explain::trace_plan(expr.as_inner());
374+
375+
Ok::<_, OptimizerError>(expr.0)
376+
}
377+
362378
macro_rules! trace_plan {
363379
(at: $span:literal, $plan:expr) => {
364380
tracing::debug_span!(target: "optimizer", $span).in_scope(|| {

src/adapter/src/optimize/view.rs

+25-5
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,13 @@
77
// the Business Source License, use of this software will be governed
88
// by the Apache License, Version 2.0.
99

10-
//! Optimizer implementation for `CREATE VIEW` statements.
10+
//! An Optimizer that
11+
//! 1. Optimistically calls `optimize_mir_constant`.
12+
//! 2. Then, if we haven't arrived at a constant, it calls `optimize_mir_local`, i.e., the
13+
//! logical optimizer.
14+
//!
15+
//! This is used for `CREATE VIEW` statements and in various other situations where no physical
16+
//! optimization is needed, such as for `INSERT` statements.
1117
1218
use std::time::Instant;
1319

@@ -18,7 +24,10 @@ use mz_transform::dataflow::DataflowMetainfo;
1824
use mz_transform::typecheck::{empty_context, SharedContext as TypecheckContext};
1925
use mz_transform::TransformCtx;
2026

21-
use crate::optimize::{optimize_mir_local, trace_plan, Optimize, OptimizerConfig, OptimizerError};
27+
use crate::optimize::{
28+
optimize_mir_constant, optimize_mir_local, trace_plan, Optimize, OptimizerConfig,
29+
OptimizerError,
30+
};
2231

2332
pub struct Optimizer {
2433
/// A typechecking context to use throughout the optimizer pipeline.
@@ -52,13 +61,24 @@ impl Optimize<HirRelationExpr> for Optimizer {
5261
trace_plan!(at: "raw", &expr);
5362

5463
// HIR ⇒ MIR lowering and decorrelation
55-
let expr = expr.lower(&self.config, self.metrics.as_ref())?;
64+
let mut expr = expr.lower(&self.config, self.metrics.as_ref())?;
5665

57-
// MIR ⇒ MIR optimization (local)
5866
let mut df_meta = DataflowMetainfo::default();
5967
let mut transform_ctx =
6068
TransformCtx::local(&self.config.features, &self.typecheck_ctx, &mut df_meta);
61-
let expr = optimize_mir_local(expr, &mut transform_ctx)?;
69+
70+
// First, we run a very simple optimizer pipeline, which only folds constants. This takes
71+
// care of constant INSERTs. (This optimizer is also used for INSERTs, not just VIEWs.)
72+
expr = optimize_mir_constant(expr, &mut transform_ctx)?;
73+
74+
// MIR ⇒ MIR optimization (local)
75+
let expr = if expr.as_const().is_some() {
76+
// No need to optimize further, because we already have a constant.
77+
OptimizedMirRelationExpr(expr)
78+
} else {
79+
// Call the real optimization.
80+
optimize_mir_local(expr, &mut transform_ctx)?
81+
};
6282

6383
if let Some(metrics) = &self.metrics {
6484
metrics.observe_e2e_optimization_time("view", time.elapsed());

src/transform/src/lib.rs

+14-3
Original file line numberDiff line numberDiff line change
@@ -552,9 +552,11 @@ pub fn fuse_and_collapse_fixpoint() -> Fixpoint {
552552
}
553553
}
554554

555-
/// Does constant folding idempotently. This needs to call `FoldConstants` together with
556-
/// `NormalizeLets` in a fixpoint loop, because currently `FoldConstants` doesn't inline CTEs, so
557-
/// these two need to alternate until fixpoint.
555+
/// Does constant folding to a fixpoint: An expression all of whose leaves are constants, of size
556+
/// small enough to be inlined and folded should reach a single `MirRelationExpr::Constant`.
557+
///
558+
/// This needs to call `FoldConstants` together with `NormalizeLets` in a fixpoint loop, because
559+
/// currently `FoldConstants` doesn't inline CTEs, so these two need to alternate until fixpoint.
558560
///
559561
/// Also note that `FoldConstants` can break the normalized form by removing all references to a
560562
/// Let.
@@ -819,6 +821,15 @@ impl Optimizer {
819821
}
820822
}
821823

824+
/// Builds a tiny optimizer, which just folds constants. For more details, see
825+
/// [fold_constants_fixpoint].
826+
pub fn constant_optimizer(_ctx: &mut TransformCtx) -> Self {
827+
Self {
828+
name: "fast_path_optimizer",
829+
transforms: vec![Box::new(fold_constants_fixpoint())],
830+
}
831+
}
832+
822833
/// Optimizes the supplied relation expression.
823834
///
824835
/// These optimizations are performed with no information about available arrangements,

0 commit comments

Comments
 (0)