Skip to content
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

feat: metadata columns #14057

Open
wants to merge 7 commits into
base: main
Choose a base branch
from

Conversation

chenkovsky
Copy link

Which issue does this PR close?

Closes #13975.

Rationale for this change

many databases support pseudo columns, for example, file_path, file_name, file_size, rowid.
for pseudo columns, we don't want to get them by default, but we want to be able to use them explicitly.

for the database that supports rowid. select * from tb won't return rowid. but we can get rowid by select rowid, * from tb. spark has already supported metadata columns. this PR want to support it in datafusion.

What changes are included in this PR?

  • add an API in table provider that will return metadata column schema.
  • change DFSchema add metadata column.
  • change logical plan e.g. TableScan to support it.

Are these changes tested?

Unit test is added

Are there any user-facing changes?

No

For FFI table provider API, one function that returns metadata column is added.

@github-actions github-actions bot added logical-expr Logical plan and expressions optimizer Optimizer rules core Core DataFusion crate catalog Related to the catalog crate common Related to common crate labels Jan 9, 2025
return metadata.qualified_field(i - self.inner.len());
}
}
self.inner.qualified_field(i)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it better not to mix inner field and meta field?

maybe we need another method meta_field(&self, i: usize)

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

actually implementing another method was my first attempt. but I found that I need to change a lot of code, because column index is used everywhere. that's why in currently implementation metadata column has index + len(fields).

Copy link
Contributor

@jayzhan211 jayzhan211 Jan 11, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't only where you need meta columns you need to change the code with meta_field? Others code that call with field remain the same.

The downside of the current approach is that whenever the schema is changed, the index of meta columns need to adjust too. I think this is error prone. Minimize the dependency of meta schema and schema is better

Copy link
Author

@chenkovsky chenkovsky Jan 11, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see. it's error prone. Can we change the offsets of metadata columns, e.g. (-1 as usize) (-2 as usize) then there's no such problem. I see some databases use this trick.

Isn't only where you need meta columns you need to change the code with meta_field? Others code that call with field remain the same.

yes, we can. but many apis use Vec to represent columns. I have to change many structs and method defnitions to pass extra parameters.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(-1 as usize) how does this large offset work? We have vector instead of map

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi @jayzhan211 I pushed a commit, could you please review it again?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay this approach looks good to me.

.collect()
let mut fields: Vec<&Field> = self.inner.fields_with_unqualified_name(name);
if let Some(schema) = self.metadata_schema() {
fields.append(&mut schema.fields_with_unqualified_name(name));
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
fields.append(&mut schema.fields_with_unqualified_name(name));
fields.append(schema.fields_with_unqualified_name(name));

let mut fields: Vec<(Option<&TableReference>, &Field)> =
self.inner.qualified_fields_with_unqualified_name(name);
if let Some(schema) = self.metadata_schema() {
fields.append(&mut schema.qualified_fields_with_unqualified_name(name));
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
fields.append(&mut schema.qualified_fields_with_unqualified_name(name));
fields.append(schema.qualified_fields_with_unqualified_name(name));

return (
Some(table_name.clone()),
Arc::new(
metadata.field(*i - METADATA_OFFSET).clone(),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

handle where i < METADATA_OFFSET

Copy link
Contributor

@jayzhan211 jayzhan211 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great, wait others to review this

Copy link
Contributor

@alamb alamb left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you @chenkovsky and @jayzhan211 -- this is a neat feature and I think has also been asked for before 💯

Also, I think the code is well structured and tested.

Before we merge this PR I think we need

  1. a test for more than one metadata column
  2. ensure this doesn't slow down planning (I will run benchmarks and report back)

I would strongly recommend we do in this PR (but could do as a follow on)

  1. More documentation (to help others and our future selves use it)
  2. Change the test to use assert_batches_eq

&self.inner.schema
}

pub fn with_metadata_schema(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we please document these APIs

@@ -55,6 +55,11 @@ pub trait TableProvider: Debug + Sync + Send {
/// Get a reference to the schema for this table
fn schema(&self) -> SchemaRef;

/// Get metadata columns of this table.
fn metadata_columns(&self) -> Option<SchemaRef> {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you please document this better -- specifically:

  1. A link to the prior art (spark metadata columns)
  2. A brief summary of what metadata columns are used for and an example (you can copy the content from the spark docs)

metadata: Option<QualifiedSchema>,
}

pub const METADATA_OFFSET: usize = usize::MAX >> 1;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you please document what this is and how it relates to DFSchema::inner

inner: QualifiedSchema,
/// Stores functional dependencies in the schema.
functional_dependencies: FunctionalDependencies,
/// metadata columns
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you provide more documentation here to document what these are (perhaps adding a link to the higher level description you write on TableProvider::metadata_columns)

pub const METADATA_OFFSET: usize = usize::MAX >> 1;

#[derive(Debug, Clone, PartialEq, Eq)]
pub struct QualifiedSchema {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please document what this struct is used for

}
}

pub fn metadata_schema(&self) -> &Option<QualifiedSchema> {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add documentation -- imagine you are someone using this API and are not familar with metadata_schema or the content of this API. I think you would want a short summary of what this is and then a link to the full details

use datafusion_common::METADATA_OFFSET;
use itertools::Itertools;

/// A User, with an id and a bank account
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is is actually quite a cool example of using metadata index

Eventually I think it would be great to add an example in https://github.com/apache/datafusion/tree/main/datafusion-examples

.unwrap();
let batch = concat_batches(&all_batchs[0].schema(), &all_batchs).unwrap();
assert_eq!(batch.num_rows(), 2);
let serializer = CsvSerializer::new().with_header(false);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To check the results, can you please use assert_batches_eq instead of converting to CSV?

That is

  1. more consistent with the rest of the codebase
  2. easier to read
  3. easier to update

For example:

let expected = vec![
"+----+----+",
"| c1 | c2 |",
"+----+----+",
"| 1 | 1 |",
"| 1 | 2 |",
"| 1 | 3 |",
"| 1 | 4 |",
"| 1 | 5 |",
"| 1 | 6 |",
"| 1 | 7 |",
"| 1 | 8 |",
"| 1 | 9 |",
"| 1 | 10 |",
"| 2 | 1 |",
"| 2 | 2 |",
"| 2 | 3 |",
"| 2 | 4 |",
"| 2 | 5 |",
"| 2 | 6 |",
"| 2 | 7 |",
"| 2 | 8 |",
"| 2 | 9 |",
"| 2 | 10 |",
"+----+----+",
];
assert_batches_sorted_eq!(expected, &results);

let all_batchs = df5.collect().await.unwrap();
let batch = concat_batches(&all_batchs[0].schema(), &all_batchs).unwrap();
let bytes = serializer.serialize(batch, true).unwrap();
assert_eq!(bytes, "1,2\n");
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we please also add a test for more than one metadata column?

@alamb
Copy link
Contributor

alamb commented Jan 12, 2025

Something other people have asked for in the past (whihc I can't find now) is the ability to know what file a particular row came from in a listing table that combines multiple files

Update: I found it at #8906

To be clear I think this PR would enable selecting a subset of files, as described on #8906 (comment)

@adriangb
Copy link
Contributor

We want this as well to hide "special" internal columns we create to speed up JSON columns. +1 for the feature!

@adriangb
Copy link
Contributor

My only question is if "metadata" is the right name for these columns? Could it be "system" columns or something like that?

@Omega359
Copy link
Contributor

Metadata column is the name I'm familiar with in other systems. For example, spark/databricks

@adriangb
Copy link
Contributor

I guess the naming doesn't really hurt our use case so okay let's go with that if it means something in the domain in general 👍🏻

@alamb
Copy link
Contributor

alamb commented Jan 13, 2025

FWIW I ran the planning benchmarks on this branch and see no measurable difference. ✅

++ critcmp main feature_metadata_columns
group                            feature_metadata_columns               main
-----                            ------------------------               ----
logical_select_all_from_1000     1.00      5.2±0.03ms        ? ?/sec    1.01      5.3±0.04ms        ? ?/sec
physical_plan_clickbench_all     1.00    226.7±1.82ms        ? ?/sec    1.00    227.1±1.40ms        ? ?/sec
physical_plan_tpcds_all          1.00   1380.0±3.29ms        ? ?/sec    1.00   1378.8±5.32ms        ? ?/sec
physical_plan_tpch_all           1.00     90.2±0.70ms        ? ?/sec    1.01     91.1±1.32ms        ? ?/sec
physical_select_all_from_1000    1.02     42.4±0.30ms        ? ?/sec    1.00     41.5±0.16ms        ? ?/sec

@berkaysynnada
Copy link
Contributor

Can these metadata columns utilize normal column properties, like ordering equivalences, constantness, distinctness etc.? For example, AFAIU rowid is an ordered column, and if I sort the table by rowid, the SortExec would be removed? (it seems to me not yet at this point) Can we iterate over the design to support those capabilities, too?

@alamb
Copy link
Contributor

alamb commented Jan 15, 2025

Can these metadata columns utilize normal column properties, like ordering equivalences, constantness, distinctness etc.? For example, AFAIU rowid is an ordered column, and if I sort the table by rowid, the SortExec would be removed? (it seems to me not yet at this point) Can we iterate over the design to support those capabilities, too?

I with this PR a custom table provider that was ordered by row_id could communicate that information to avoid a SortExec

From what I can tell, the metadata columns is only a notion in the LogicalPlan

Specifically, the ExecutionPlan returned by the provider is no different than any other ExecutionPlan so it can communicate sortedness via ExecutionPlan::properties as normal

@berkaysynnada
Copy link
Contributor

Can these metadata columns utilize normal column properties, like ordering equivalences, constantness, distinctness etc.? For example, AFAIU rowid is an ordered column, and if I sort the table by rowid, the SortExec would be removed? (it seems to me not yet at this point) Can we iterate over the design to support those capabilities, too?

I with this PR a custom table provider that was ordered by row_id could communicate that information to avoid a SortExec

From what I can tell, the metadata columns is only a notion in the LogicalPlan

Specifically, the ExecutionPlan returned by the provider is no different than any other ExecutionPlan so it can communicate sortedness via ExecutionPlan::properties as normal

What I mean is:

https://github.com/chenkovsky/datafusion/blob/5c4b5c4c7aee47b6287e5fcf32d87485ee1c9e37/datafusion/core/tests/sql/metadata_columns.rs#L389

When I print this query, there exists a SortExec for _rowid. But what I understand is _rowid should be a one-by-one increasing column?

@chenkovsky
Copy link
Author

Can these metadata columns utilize normal column properties, like ordering equivalences, constantness, distinctness etc.? For example, AFAIU rowid is an ordered column, and if I sort the table by rowid, the SortExec would be removed? (it seems to me not yet at this point) Can we iterate over the design to support those capabilities, too?

I with this PR a custom table provider that was ordered by row_id could communicate that information to avoid a SortExec
From what I can tell, the metadata columns is only a notion in the LogicalPlan
Specifically, the ExecutionPlan returned by the provider is no different than any other ExecutionPlan so it can communicate sortedness via ExecutionPlan::properties as normal

What I mean is:

https://github.com/chenkovsky/datafusion/blob/5c4b5c4c7aee47b6287e5fcf32d87485ee1c9e37/datafusion/core/tests/sql/metadata_columns.rs#L389

When I print this query, there exists a SortExec for _rowid. But what I understand is _rowid should be a one-by-one increasing column?

Maybe not, I use vec to store values in test, but if the inner datastructure is btree, the scan order is not always increasing.

@alamb
Copy link
Contributor

alamb commented Jan 16, 2025

I plan to merge this PR in tomorrow and file follow on tickets unless anyone would like additional time to review

@alamb
Copy link
Contributor

alamb commented Jan 18, 2025

BTW I am waiting to merge this PR until I have time to file follow on tickets / docs / etc as I suggested above

If anyone needs it sooner then I think we can merge it in and do the follow ons later

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
catalog Related to the catalog crate common Related to common crate core Core DataFusion crate logical-expr Logical plan and expressions optimizer Optimizer rules
Projects
None yet
Development

Successfully merging this pull request may close these issues.

metadata column support
6 participants