Skip to content

Commit

Permalink
feat: use custom table provider to reuse statistics. (parseablehq#566)
Browse files Browse the repository at this point in the history
This PR adds a new table provider implementation that directly 
creates physical plan from manifest file entries. File level 
statistics are aggregated into table level statistics which is 
available for count optimisation.

Note: Aggregate Statistics will only take effect for time 
range which lie exactly on minute boundary. Otherwise 
they are treated as partial filters and plan changes, 
hence will not use statistics anymore.
---------

Co-authored-by: Nitish Tiwari <[email protected]>
  • Loading branch information
trueleo and nitisht authored Dec 11, 2023
1 parent 40cddb5 commit c6f1f74
Show file tree
Hide file tree
Showing 6 changed files with 417 additions and 218 deletions.
2 changes: 1 addition & 1 deletion server/src/catalog.rs
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ pub mod snapshot;
pub use manifest::create_from_parquet_file;

pub trait Snapshot {
fn manifests(&self, time_predicates: Vec<PartialTimeFilter>) -> Vec<ManifestItem>;
fn manifests(&self, time_predicates: &[PartialTimeFilter]) -> Vec<ManifestItem>;
}

pub trait ManifestFile {
Expand Down
2 changes: 1 addition & 1 deletion server/src/catalog/snapshot.rs
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ impl Default for Snapshot {
}

impl super::Snapshot for Snapshot {
fn manifests(&self, time_predicates: Vec<PartialTimeFilter>) -> Vec<ManifestItem> {
fn manifests(&self, time_predicates: &[PartialTimeFilter]) -> Vec<ManifestItem> {
let mut manifests = self.manifest_list.clone();
for predicate in time_predicates {
match predicate {
Expand Down
12 changes: 5 additions & 7 deletions server/src/query.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
*/

mod filter_optimizer;
mod listing_table_builder;
mod stream_schema_provider;
mod table_provider;

Expand All @@ -34,7 +35,6 @@ use datafusion::prelude::*;
use itertools::Itertools;
use once_cell::sync::Lazy;
use std::collections::HashMap;
use std::ops::Add;
use std::path::{Path, PathBuf};
use std::sync::Arc;
use sysinfo::{System, SystemExt};
Expand Down Expand Up @@ -172,12 +172,10 @@ impl Query {
PartialTimeFilter::Low(std::ops::Bound::Included(self.start.naive_utc())).binary_expr(
Expr::Column(Column::from_name(event::DEFAULT_TIMESTAMP_KEY)),
);
let end_time_filter = PartialTimeFilter::High(std::ops::Bound::Excluded(
self.end.add(chrono::Duration::milliseconds(1)).naive_utc(),
))
.binary_expr(Expr::Column(Column::from_name(
event::DEFAULT_TIMESTAMP_KEY,
)));
let end_time_filter =
PartialTimeFilter::High(std::ops::Bound::Excluded(self.end.naive_utc())).binary_expr(
Expr::Column(Column::from_name(event::DEFAULT_TIMESTAMP_KEY)),
);

// see https://github.com/apache/arrow-datafusion/pull/8400
// this can be eliminated in later version of datafusion but with slight caveat
Expand Down
206 changes: 206 additions & 0 deletions server/src/query/listing_table_builder.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,206 @@
/*
* Parseable Server (C) 2022 - 2023 Parseable, Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/

use std::{collections::HashMap, ops::Bound, pin::Pin, sync::Arc};

use arrow_schema::Schema;
use datafusion::{
datasource::{
file_format::parquet::ParquetFormat,
listing::{ListingOptions, ListingTable, ListingTableConfig, ListingTableUrl},
},
error::DataFusionError,
logical_expr::col,
};
use futures_util::{future, stream::FuturesUnordered, Future, TryStreamExt};
use itertools::Itertools;
use object_store::{ObjectMeta, ObjectStore};

use crate::{
event::DEFAULT_TIMESTAMP_KEY,
storage::{ObjectStorage, OBJECT_STORE_DATA_GRANULARITY},
utils::TimePeriod,
};

use super::PartialTimeFilter;

// Listing Table Builder for querying old data
#[derive(Debug, Default)]
pub struct ListingTableBuilder {
stream: String,
listing: Vec<String>,
}

impl ListingTableBuilder {
pub fn new(stream: String) -> Self {
Self {
stream,
..Self::default()
}
}

pub async fn populate_via_listing(
self,
storage: Arc<dyn ObjectStorage + Send>,
client: Arc<dyn ObjectStore>,
time_filters: &[PartialTimeFilter],
) -> Result<Self, DataFusionError> {
let start_time = time_filters
.iter()
.filter_map(|x| match x {
PartialTimeFilter::Low(Bound::Excluded(x)) => Some(x),
PartialTimeFilter::Low(Bound::Included(x)) => Some(x),
_ => None,
})
.min()
.cloned();

let end_time = time_filters
.iter()
.filter_map(|x| match x {
PartialTimeFilter::High(Bound::Excluded(x)) => Some(x),
PartialTimeFilter::High(Bound::Included(x)) => Some(x),
_ => None,
})
.max()
.cloned();

let Some((start_time, end_time)) = start_time.zip(end_time) else {
return Err(DataFusionError::NotImplemented(
"The time predicate is not supported because of possibly querying older data."
.to_string(),
));
};

let prefixes = TimePeriod::new(
start_time.and_utc(),
end_time.and_utc(),
OBJECT_STORE_DATA_GRANULARITY,
)
.generate_prefixes();

let prefixes = prefixes
.into_iter()
.map(|entry| {
let path =
relative_path::RelativePathBuf::from(format!("{}/{}", &self.stream, entry));
storage.absolute_url(path.as_relative_path()).to_string()
})
.collect_vec();

let mut minute_resolve: HashMap<String, Vec<String>> = HashMap::new();
let mut all_resolve = Vec::new();

for prefix in prefixes {
let components = prefix.split_terminator('/');
if components.last().is_some_and(|x| x.starts_with("minute")) {
let hour_prefix = &prefix[0..prefix.rfind("minute").expect("minute exists")];
minute_resolve
.entry(hour_prefix.to_owned())
.and_modify(|list| list.push(prefix))
.or_default();
} else {
all_resolve.push(prefix)
}
}

type ResolveFuture = Pin<
Box<dyn Future<Output = Result<Vec<ObjectMeta>, object_store::Error>> + Send + 'static>,
>;

let tasks: FuturesUnordered<ResolveFuture> = FuturesUnordered::new();

for (listing_prefix, prefix) in minute_resolve {
let client = Arc::clone(&client);
tasks.push(Box::pin(async move {
let mut list = client
.list(Some(&object_store::path::Path::from(listing_prefix)))
.await?
.try_collect::<Vec<_>>()
.await?;

list.retain(|object| {
prefix.iter().any(|prefix| {
object
.location
.prefix_matches(&object_store::path::Path::from(prefix.as_ref()))
})
});

Ok(list)
}));
}

for prefix in all_resolve {
let client = Arc::clone(&client);
tasks.push(Box::pin(async move {
client
.list(Some(&object_store::path::Path::from(prefix)))
.await?
.try_collect::<Vec<_>>()
.await
.map_err(Into::into)
}));
}

let res: Vec<Vec<String>> = tasks
.and_then(|res| {
future::ok(
res.into_iter()
.map(|res| res.location.to_string())
.collect_vec(),
)
})
.try_collect()
.await?;

let mut res = res.into_iter().flatten().collect_vec();
res.sort();
res.reverse();

Ok(Self {
stream: self.stream,
listing: res,
})
}

pub fn build(
self,
schema: Arc<Schema>,
map: impl Fn(Vec<String>) -> Vec<ListingTableUrl>,
) -> Result<Option<Arc<ListingTable>>, DataFusionError> {
if self.listing.is_empty() {
return Ok(None);
}

let file_format = ParquetFormat::default().with_enable_pruning(Some(true));
let file_sort_order = vec![vec![col(DEFAULT_TIMESTAMP_KEY).sort(true, false)]];
let listing_options = ListingOptions::new(Arc::new(file_format))
.with_file_extension(".parquet")
.with_file_sort_order(file_sort_order)
.with_collect_stat(true)
.with_target_partitions(1);

let config = ListingTableConfig::new_with_multi_paths(map(self.listing))
.with_listing_options(listing_options)
.with_schema(schema);

let listing_table = Arc::new(ListingTable::try_new(config)?);
Ok(Some(listing_table))
}
}
Loading

0 comments on commit c6f1f74

Please sign in to comment.