use std::any::Any;
use std::collections::{BTreeMap, BTreeSet};
use std::rc::{Rc, Weak};
use differential_dataflow::Collection;
use mz_compute_types::sinks::{ComputeSinkConnection, ComputeSinkDesc};
use mz_expr::{permutation_for_arrangement, EvalError, MapFilterProject};
use mz_ore::soft_assert_or_log;
use mz_ore::str::StrExt;
use mz_ore::vec::PartialOrdVecExt;
use mz_repr::{Diff, GlobalId, Row};
use mz_storage_types::controller::CollectionMetadata;
use mz_storage_types::errors::DataflowError;
use mz_timely_util::operator::CollectionExt;
use timely::container::CapacityContainerBuilder;
use timely::dataflow::scopes::Child;
use timely::dataflow::Scope;
use timely::progress::Antichain;
use crate::compute_state::SinkToken;
use crate::logging::compute::LogDataflowErrors;
use crate::render::context::Context;
use crate::render::{RenderTimestamp, StartSignal};
impl<'g, G, T> Context<Child<'g, G, T>>
where
G: Scope<Timestamp = mz_repr::Timestamp>,
T: RenderTimestamp,
{
pub(crate) fn export_sink(
&self,
compute_state: &mut crate::compute_state::ComputeState,
tokens: &BTreeMap<GlobalId, Rc<dyn std::any::Any>>,
dependency_ids: BTreeSet<GlobalId>,
sink_id: GlobalId,
sink: &ComputeSinkDesc<CollectionMetadata>,
start_signal: StartSignal,
ct_times: Option<Collection<G, (), Diff>>,
) {
soft_assert_or_log!(
sink.non_null_assertions.is_strictly_sorted(),
"non-null assertions not sorted"
);
let mut needed_tokens = Vec::new();
for dep_id in dependency_ids {
if let Some(token) = tokens.get(&dep_id) {
needed_tokens.push(Rc::clone(token))
}
}
let bundle = self
.lookup_id(mz_expr::Id::Global(sink.from))
.expect("Sink source collection not loaded");
let (ok_collection, mut err_collection) = if let Some(collection) = &bundle.collection {
collection.clone()
} else {
let (key, _arrangement) = bundle
.arranged
.iter()
.next()
.expect("Invariant violated: at least one collection must be present.");
let unthinned_arity = sink.from_desc.arity();
let (permutation, thinning) = permutation_for_arrangement(key, unthinned_arity);
let mut mfp = MapFilterProject::new(unthinned_arity);
mfp.permute_fn(|c| permutation[c], thinning.len() + key.len());
bundle.as_collection_core(mfp, Some((key.clone(), None)), self.until.clone())
};
if let Some(logger) = compute_state.compute_logger.clone() {
err_collection = err_collection.log_dataflow_errors(logger, sink_id);
}
let mut ok_collection = ok_collection.leave();
let mut err_collection = err_collection.leave();
if let Some(&expiration) = self.dataflow_expiration.as_option() {
let token = Rc::new(());
let shutdown_token = Rc::downgrade(&token);
ok_collection = ok_collection.expire_collection_at(
&format!("{}_export_sink_oks", self.debug_name),
expiration,
Weak::clone(&shutdown_token),
);
err_collection = err_collection.expire_collection_at(
&format!("{}_export_sink_errs", self.debug_name),
expiration,
shutdown_token,
);
needed_tokens.push(token);
}
let non_null_assertions = sink.non_null_assertions.clone();
let from_desc = sink.from_desc.clone();
if !non_null_assertions.is_empty() {
let name = format!("NullAssertions({sink_id:?})");
type CB<C> = CapacityContainerBuilder<C>;
let (oks, null_errs) =
ok_collection.map_fallible::<CB<_>, CB<_>, _, _, _>(&name, move |row| {
let mut idx = 0;
let mut iter = row.iter();
for &i in &non_null_assertions {
let skip = i - idx;
let datum = iter.nth(skip).unwrap();
idx += skip + 1;
if datum.is_null() {
return Err(DataflowError::EvalError(Box::new(
EvalError::MustNotBeNull(
format!("column {}", from_desc.get_name(i).as_str().quoted())
.into(),
),
)));
}
}
Ok(row)
});
ok_collection = oks;
err_collection = err_collection.concat(&null_errs);
}
let region_name = match sink.connection {
ComputeSinkConnection::Subscribe(_) => format!("SubscribeSink({:?})", sink_id),
ComputeSinkConnection::MaterializedView(_) => {
format!("MaterializedViewSink({:?})", sink_id)
}
ComputeSinkConnection::ContinualTask(_) => {
format!("ContinualTaskSink({:?})", sink_id)
}
ComputeSinkConnection::CopyToS3Oneshot(_) => {
format!("CopyToS3OneshotSink({:?})", sink_id)
}
};
self.scope
.parent
.clone()
.region_named(®ion_name, |inner| {
let sink_render = get_sink_render_for::<_>(&sink.connection);
let sink_token = sink_render.render_sink(
compute_state,
sink,
sink_id,
self.as_of_frontier.clone(),
start_signal,
ok_collection.enter_region(inner),
err_collection.enter_region(inner),
ct_times.map(|x| x.enter_region(inner)),
);
if let Some(sink_token) = sink_token {
needed_tokens.push(sink_token);
}
let collection = compute_state.expect_collection_mut(sink_id);
collection.sink_token = Some(SinkToken::new(Box::new(needed_tokens)));
});
}
}
pub(crate) trait SinkRender<G>
where
G: Scope<Timestamp = mz_repr::Timestamp>,
{
fn render_sink(
&self,
compute_state: &mut crate::compute_state::ComputeState,
sink: &ComputeSinkDesc<CollectionMetadata>,
sink_id: GlobalId,
as_of: Antichain<mz_repr::Timestamp>,
start_signal: StartSignal,
sinked_collection: Collection<G, Row, Diff>,
err_collection: Collection<G, DataflowError, Diff>,
ct_times: Option<Collection<G, (), Diff>>,
) -> Option<Rc<dyn Any>>;
}
fn get_sink_render_for<G>(
connection: &ComputeSinkConnection<CollectionMetadata>,
) -> Box<dyn SinkRender<G>>
where
G: Scope<Timestamp = mz_repr::Timestamp>,
{
match connection {
ComputeSinkConnection::Subscribe(connection) => Box::new(connection.clone()),
ComputeSinkConnection::MaterializedView(connection) => Box::new(connection.clone()),
ComputeSinkConnection::ContinualTask(connection) => Box::new(connection.clone()),
ComputeSinkConnection::CopyToS3Oneshot(connection) => Box::new(connection.clone()),
}
}