-
Notifications
You must be signed in to change notification settings - Fork 436
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
Simplify Variant shredding and refactor for clarity #461
base: master
Are you sure you want to change the base?
Conversation
c4b435f
to
8352319
Compare
VariantShredding.md
Outdated
We extract all homogenous data items of a certain path into `typed_value`, and set aside incompatible data items in `variant_value`. | ||
Intuitively, incompatibilities within the same path may occur because we store the shredding schema per Parquet file, and each file can contain several row groups. | ||
Selecting a type for each field that is acceptable for all rows would be impractical because it would require buffering the contents of an entire file before writing. | ||
All fields for a variant, whether shredded or not, must be present in the metadata. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This may be controversial. I'm trying to say that you should not need to modify the metadata when reading. The reconstructed object should be able to use the stored metadata without adding fields.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm a little confused. When the field is not shredded, we will not have metadata for it, right? When it's getting shredded, then it will be like a column and we will generate metadata so it can be used for filtering/pruning?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@sfc-gh-aixu, this is saying that when writing, the metadata for a shredded value and the metadata for a non-shredded value should be identical. Writers should not alter the metadata by removing shredded field names so that readers do not need to rewrite the metadata (and values) to add it back.
For example, consider an event that looks like this:
{
"id": 102,
"event_type": "signup",
"event_timestamp": "2024-10-21T20:06:34.198724",
"payload": {
"a": 1,
"b": 2
}
}
And a shredding schema:
optional group event (VARIANT) {
required binary metadata;
optional binary value;
optional group typed_value {
required group event_type {
optional binary value;
optional binary typed_value (STRING);
}
required group event_timestamp {
optional binary value;
optional int64 typed_value (TIMESTAMP(true, MICROS));
}
}
}
The top-level event_type
and event_timestamp
fields are shredded. But this is saying that the Variant metadata
must include those field names. That ensure that the existing binary metadata can be returned to the engine without adding event_type
and event_timestamp
fields when merging those fields into the top-level Variant value
when the entire Variant is projected.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for detailed explanation. Later I realize this is about variant metadata
and what I was talking about was column metadata (stats).
I get what you are saying: when the entire Variant is projected, we need to reconstruct the original value
and metadata
by merging back the shredded fields if the metadata
after shredding excludes the shredded fields.
That makes sense to me to reduce the metadata reconstruction on the read side.
VariantShredding.md
Outdated
|
||
Similarly the elements of an `array` must be a group containing one or more of `object`, `array`, `typed_value` or `variant_value`. | ||
Each shredded field is represented as a required group that contains a `variant_value` and a `typed_value` field. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why each shredded field should be a required group is not clear to me. If fields were allowed to be optional, that would be another way of indicating non-existence of fields.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The primary purpose is to reduce the number of cases that implementers have to deal with. If all of the cases can be expressed with 2 optional fields rather than 2 optional fields inside an optional group, then the group should be required to simplify as much as possible.
In addition, every level in Parquet that is optional introduces another repetition/definition level. That adds up quickly with nested structures and ends up taking unnecessary space.
VariantShredding.md
Outdated
The `typed_value` field may be any type that has a corresponding Variant type. | ||
For each value in the data, at most one of the `typed_value` and `variant_value` may be non-null. | ||
A writer may omit either field, which is equivalent to all rows being null. | ||
If both fields are non-null and either is not an object, the value is invalid. Readers must either fail or return the `typed_value`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@RussellSpitzer and @gene-db, this could use some attention.
Here, if both value
and typed_value
are non-null I initially thought it made more sense to prefer value
because it doesn't need to be re-encoded and may have been coerced by an engine to the shredded type.
However, this conflicts with object fields, where the value of typed_value
is preferred so that data skipping is correct. If the object's value
could contains a field that conflicts with a sub-field's typed_value
there is no way of knowing from field stats. If we preferred the field value stored in the object's value
then data skipping could be out of sync with the value returned in the case of a conflict.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
the value
is invalid
If both fields are non-null and either is not an object, the value is invalid. Readers must either fail or return the `typed_value`. | |
If both fields are non-null and either is not an object, the `value` is invalid. Readers must either fail or return the `typed_value`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why aren't we just being proscriptive here? Isn't this essentially saying you can duplicate a subfield-field between typed_value and value? Wouldn't it be safer to just say this cannot be done?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The problem is that readers won't actually implement restrictions like this and we can't fully prevent it. It is invalid for a writer to produce a value where value
and typed_value
conflict. But writer bugs happen and readers need to know what to do when they encounter that situation. Otherwise we would get different behaviors between readers that are processing the same data file.
It all comes down to end users -- if a writer bug produces data like this, readers will implement the ability to read because the data still exists and can be recovered. When that happens, we want to know how it is interpreted.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My take is if readers have bugs and produce invalid values, I'm not sure you can really trust most of the data at all (even metadata). It sounds like we are assuming 1 specific type of bug where readers accidentally forget to clear a field?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The point is not the bug. It is that we want to make it valid to read a projection without checking the value
for bugs.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
IMHO, trying to make the projection of an object to another one without having to read the value is too spark specific. For example, with keys "a" and "b" shredded, If I am casting {"a":2, "b":3, "c":4} to a struct with keys "a" and "b", I can easily imagine a cast semantic that will fail that cast and such a semantic will force us reading both the typed_value and value unless value is null.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think I understand. If a
and b
are shredded, then there should be no fields with that name in value
and the spec is stating that you don't need to check for them. That means all of the information needed to continue is in the shredded Parquet columns. That's not specific to Spark.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What I was trying to say is, if I am casting that field to a struct with "a" and "b" but "c" exists in value, then some engines will require that cast to fail, because the object with keys "a", "b" and "c" do not match the struct schema. Apparently, with spark such a cast succeeds by producing the struct with the subset of the keys that match the struct schema.
VariantEncoding.md
Outdated
|---------------|-----------|----------------------------------------------------------|--------------------------------------| | ||
| Null type | null | `null` | `null` | | ||
| Boolean | boolean | `true` or `false` | `true` | | ||
| Exact Numeric | number | Digits in fraction must match scale, no exponent | `34`, 34.00 | |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For exact numerics, we should allow truncating trailing zeros. For example, int8
value 1
and decimal(5,2)
value 100
can both be represented as a JSON value 1
.
Also, should the example be quoted to stay consistent?
| Exact Numeric | number | Digits in fraction must match scale, no exponent | `34`, 34.00 | | |
| Exact Numeric | number | Digits in fraction must match scale, no exponent | `34`, `34.00` | |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the intent of considering Exact Numeric
to be a single logical type is that we consider the int8
value 1
to be logically equivalent to decimal(5,2)
with unscaled value 100
. If that's the case, I think we'd want the produced JSON to be the same for both (probably 1
in both cases), and not recommend having the fraction match the scale.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@gene-db, @cashmand, these are concerns for the engine layer, not for storage. If Spark wants to automatically coerce between types that's fine, but the compromise that we talked about a couple months ago was to leave this out of the shredding spec and delegate the behavior to engines. Storage should always produce the data that was stored, without modification.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, the engine should be the one concerned with changing types.
However, my original question was about this JSON representation wording. Currently, the Representation requirements
for an Exact Numeric
says the Digits in fraction must match scale
. However, because the Exact Numeric
is considered a logical type, the value 1
could be stored in the Variant as int8
1 or decimal(5,2)
100. Both of those would be the same numeric value, so we should allow truncating trailing zeros in the JSON representation, instead of requiring that the digits in the fraction match the scale.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@gene-db, the JSON representation should match the physical type as closely as possible. The reader can interpret the value however it chooses to, but a storage implementation should not discard the information.
If you want to produce 34 from 34.00 stored as decimal(9, 2)
then the engine is responsible for casting the value to int8
and then producing JSON. The JSON representation for the original decimal(9, 2)
value is 34.00
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@rdblue I am confused with this JSON chart then. If we are talking about "storage implementation", then are you expecting there is a "storage implementation" that is converting variant values to JSON? When will storage convert a variant value to a JSON string?
I originally thought this chart was trying to say, "When an engine wants to convert a variant value to a JSON string, here are the rules". Therefore, we should allow engines to cast integral decimals to integers before converting to JSON, as you already mentioned in your previous comment.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think I agree with @gene-db on this. I think any json representation that has semantically the same meaning in JSON should be allowed. Translation to JSON is inherently lossy and I think trying to match semantics will be more error prone then it is worth (i.e. it should be a non-goal to expect it to be able to reconstruct the exact same variant from the proposed JSON representation).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think maybe the wording or presentation of this mapping is a bit confusing.
I think we are on all on the same page of allowing engines to "normalize" the Variant value. For example, I think the Spark implementation already normalizes 1.00
to 1
. There are also many optimizations and efficiency aspects with normalization, so we should not disallow that.
Maybe what this chart is trying to show is: "if you want to output a Variant value as a JSON string, this is the output format you should use". So, for numbers, the conversion should be like 1
or 1.23
(no quotes), not "1"
, or "1.23"
. If this chart was about the JSON output formatting, would that be more clear?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
When an engine wants to convert a variant value to a JSON string, here are the rules
Yes, this is correct. We want a clear way to convert to a JSON string. However, the normalization needs to happen first. We don't want to specify that the JSON must be any more lossy than it already is.
Why would we require an engine to produce a normalized value?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why would we require an engine to produce a normalized value?
At least for me, I don't think it is about "requiring" and engine to produce a normalized value first. I think if an engine is reading variant and converting it to JSON, it is possibly doing so through an internal representation so it can still apply operators on top of the JSON value and possibly even storing it as an internal representation. Conversion to a string is really only an end-user visible thing. So when I read this it seems to be requiring an engine to NOT normalize which could be hard to implement for some engines.
|
||
Dictionary IDs in a `variant_value` field refer to entries in the top-level `metadata` field. | ||
If a Variant is missing in a context where a value is required, readers must either fail or return a Variant null: basic type 0 (primitive) and physical type 0 (null). | ||
For example, if a Variant is required (like `measurement` above) and both `value` and `typed_value` are null, the returned `value` must be `00` (Variant null). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As mentioned in my previous comment, I think it would be invalid for measurement
to have both value
and typed_value
be null, and should be an error. I don't understand why we're recommend returning variant null as an option.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This rule is to address the fact that arrays cannot contain a missing value. This is saying that if a value is required but both are null, the implementation must fill in a variant null.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This rule(both value being null should be interpreted as json-null) is valid only for top level variant and array elements? I wonder how a top level variant can be inserted as both value and typed_value being null if the top level field is required. That seems inconsistent. For arrays, it looks like we could also require value being variant encoded null(json null) rather than allowing both fields to be null.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@sfc-gh-saya, if the top-level field is required but both fields are null, then the reader must produce a variant null value, 00
. We must state what happens in cases like this because it is possible for writers to produce them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If the writers produce nulls for both value
and typed_value
, it's like a corrupted files and I feel it's reasonable for the readers to error out rather than give a default value.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@rdblue Shouldn't this always be an error? This looks like invalid shredded data.
Above in the chart, we say (value=null, typed_value=null) | The value is missing; only valid for shredded object fields
. This means we can only have (null, null)
for shredded object fields.
In other scenarios, shredding should never produce (null, null)
. If there was ever a required variant that is a variant-null, then the shredding scheme should produce (variant-null, null)
, and should never produce (null, null)
.
At a high level, we replace the `value` field of the Variant Parquet group with one or more fields called `object`, `array`, `typed_value`, and `variant_value`. | ||
These represent a fixed schema suitable for constructing the full Variant value for each row. | ||
For example, the query `SELECT variant_get(event, '$.event_ts', 'timestamp') FROM tbl` only needs to load field `event_ts`, and shredding can enable columnar projection that ignores the rest of the `event` Variant. | ||
Similarly, for the query `SELECT * FROM tbl WHERE variant_get(event, '$.event_type', 'string') = 'signup'`, the `event_type` shredded column metadata can be used for skipping and to lazily load the rest of the Variant. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Similarly, for the query `SELECT * FROM tbl WHERE variant_get(event, '$.event_type', 'string') = 'signup'`, the `event_type` shredded column metadata can be used for skipping and to lazily load the rest of the Variant. | |
Similarly, for the query `SELECT * FROM tbl WHERE variant_get(event, '$.event_type', 'string') = 'signup'`, the `event_type` shredded column metadata can be used for skipping while the rest of the Variant is lazily loaded for matching pages. |
# Data Skipping | ||
All elements of an array must be non-null because `array` elements in a Variant cannot be missing. | ||
That is, either `typed_value` or `value` (but not both) must be non-null. | ||
Null elements must be encoded in `value` as Variant null: basic type 0 (primitive) and physical type 0 (null). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
just for consistency it was written as
`00` (Variant null).
Earlier in the doc but this is fine too
VariantShredding.md
Outdated
| `{"error_msg": "malformed: ..."}` | `{"error_msg", "malformed: ..."}` | null | | | | | Object with no shredding | | ||
| `"malformed: not an object"` | `malformed: not an object` | null | | | | | Not an object (stored as Variant string) | | ||
| `{"event_ts": 1729794240241, "click": "_button"}` | `{"click": "_button"}` | non-null | null | null | null | 1729794240241 | Field `event_type` is missing | | ||
| `{"event_type": null, "event_ts": 1729794954163}` | null | non-null | `00` (field exists, is null) | null | null | 1729794954163 | Field `event_type` is present and is null | |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some more requested examples,
Could we have where "event_ts" is a Date or something non transformable into a timestamp?
I assume this would make value be {"event_ts": "08-03-2025"}
while typed_value would be null
I also wonder if we could do a single example for a doubly nested field showing where typed_value.address.value != null. All the examples here cover a primitive field being typed, so It may be nice to show the behavior with a object being typed.
{
Name
Address {
City
ZIP (Shredded as INT but some values as String?)
}
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I added most other examples, but I don't think that we need the nested example because it would make the table much larger. I also cover nesting in the next section specifically.
VariantShredding.md
Outdated
|
||
The `typed_value` associated with any Variant `value` field can be any shredded type according to the rules above. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think I understand this sentence, but I believe I understand the intent is that you can have objects or elements within arrays also shredded?
I think the tables above are easier for me to follow than the parquet schema below. I understand though if that's difficult to depict.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is just saying that any time you have a value
field, you can also have a typed_value
field that might be any shredded type, like an array nested in a field or an object nested in an array.
|
||
Consider the following example: | ||
Statistics for `typed_value` columns can be used for file, row group, or page skipping when `value` is always null (missing). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we need to specify "null" vs "variant null" I get a little confused sometimes in the doc.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Isn't "null (missing)" clear that it is not variant null? Missing is only used to mean one form of null in the text.
“not an object” | ||
] | ||
``` | ||
When the corresponding `value` column is all nulls, all values must be the shredded `typed_value` field's type. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sometimes we refer to the value
as a column and sometimes as a field. Just wondering if we should take a pass to standardize unless there is another meaning i'm not following here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think I was referring to "value" as a Parquet column. All object variant fields should be referred to as "field".
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
it might be good to have this in a glossary at the top.
VariantShredding.md
Outdated
By the same token, `variant_value` may be absent, which is equivalent to their value being always null (in which case the field will always have the value Null or have the type of the `typed_value` column). | ||
| Variant Type | Equivalent Parquet Type | | ||
|-----------------------------|------------------------------| | ||
| boolean | BOOLEAN | |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this goes back to earlier debates about allowed lossiness at the storage layer but it seems we might lose a lot of benefits to shredding if a single variant has multiple exact integer representations (e.g. mix of int32 and int64)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the state of this is that the storage will not do any conversions at all. However, the engine itself is allowed to "normalize" variants to optimize. In this case, engines will probably normalize within "exact numerics" in order to make shredding more effective.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As I digest the types, I feel like the concept of exact numeric is purely an engine concern and is an entirely new concept for parquet that isnt really explicitly modelled anyplace in the spec. Given this I think my preferences would be:
- Remove the concept of this type of equality from the parquet spec and leave it in the engine.
- Make conversion to JSON an appendix as a recommendation
- Make sure we design the variant in parquet to optimized with shredding independent from the engine (I think this means adding a type identifier discriminant field for the shredded column) also as I stated in another comment, it would be good to ensure the bit order for values makes it so that we can make the widest use of stats (e.g. type info is in the most significant bits) for the non-shredded value (iiuc this might require a second version for the non-shredded value, depending on what the current bit order means)
Thoughts?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The intent is to leave normalization and conversion to engines, but I don't think that we need to remove the Variant logical types in order to do that.
I also think that we need to have a reliable JSON representation or else we will get different behavior and conventions across engines. We want people to be able to rely on what the JSON means, even if it is necessarily lossy in some cases.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The intent is to leave normalization and conversion to engines, but I don't think that we need to remove the Variant logical types in order to do that.
I think this might just be a philosophical difference of opinion. But I believe Parquet implementations should have the flexibility to appropriately optimize across physical type.
I think we should either:
- Not have the notion of logical type correspondence as part of variant in parquet. Under this assumption we should provide enough modeling that parquet can do a reasonable job shredding and reproducing the exact physical types after shredding. Parquet is used in a lot of contexts and I think in many cases the engines might not be very sophisticate here.
- Not require parquet to reproduce the exact physical types when reading back values (i.e. allow it to normalization).
Both approaches have there strengths, but what I would like to avoid is a tight coupling between the engine and parquet libraries to achieve efficient storage of variants.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The original motivation for introducing the "logical type" for the Variant binary encoding was to separate the "logical" data value, from the "physical" storage of the value, so that optimizations may be performed without altering the "logical" value. The value is important, not the storage type. For example, it shouldn't matter if the string "hello"
is stored as a short-string or a string, and it shouldn't matter if the integer 1
is stored as an int8
or int16
. Within the same "logical type" the equivalent value can be stored in different ways.
We haven't fully defined the equality semantics yet, but the goal for the logical types is that if we had 2 Variant values:
"hello"
(as a short-string)"hello"
(as a string)
those Variant values are considered equivalent, even though they are physically encoded differently.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@gene-db This makes sense. I think we need to decide at the parquet level to either:
- Use this definition of logical type and make it valid for parquet to do normalization (e.g. physical types put in are not necessarily physical types you get out).
- Make it possible for parquet to do lossless normalization (e.g. store values in the widest type that makes sense, and then also store metadata to retrieve types).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Use this definition of logical type and make it valid for parquet to do normalization (e.g. physical types put in are not necessarily physical types you get out).
My opinion is that parquet the format should not "do" normalization (though it would be fine if the writer implementations did)
I think the state of this is that the storage will not do any conversions at all. However, the engine itself is allowed to "normalize" variants to optimize.
I believe this writeup from @gene-db describes what I would expect from a formrat. In summary:
- The spec itself doesn't require / permit any normalization of values
- The implementations of writers (e.g. parquet-rs or java-rs) could have options to automatically coerce types
My rationale is that one normalization scheme is unlikely to work for all usecases, so mandating something in the spec will unnecessarily constrain users
@etseidl, @dsgibbons and others have been heading this way in parquet-rs recently (for example apache/arrow-rs#6828) when normalizing data / schema might be preferred for wider compatibility, but users can still choose to use the raw data and maintain lossless roundtrip)
VariantEncoding.md
Outdated
|---------------|-----------|----------------------------------------------------------|--------------------------------------| | ||
| Null type | null | `null` | `null` | | ||
| Boolean | boolean | `true` or `false` | `true` | | ||
| Exact Numeric | number | Digits in fraction must match scale, no exponent | `34`, 34.00 | |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think maybe the wording or presentation of this mapping is a bit confusing.
I think we are on all on the same page of allowing engines to "normalize" the Variant value. For example, I think the Spark implementation already normalizes 1.00
to 1
. There are also many optimizations and efficiency aspects with normalization, so we should not disallow that.
Maybe what this chart is trying to show is: "if you want to output a Variant value as a JSON string, this is the output format you should use". So, for numbers, the conversion should be like 1
or 1.23
(no quotes), not "1"
, or "1.23"
. If this chart was about the JSON output formatting, would that be more clear?
VariantShredding.md
Outdated
By the same token, `variant_value` may be absent, which is equivalent to their value being always null (in which case the field will always have the value Null or have the type of the `typed_value` column). | ||
| Variant Type | Equivalent Parquet Type | | ||
|-----------------------------|------------------------------| | ||
| boolean | BOOLEAN | |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the state of this is that the storage will not do any conversions at all. However, the engine itself is allowed to "normalize" variants to optimize. In this case, engines will probably normalize within "exact numerics" in order to make shredding more effective.
optional group shredded_variant_name (VARIANT) { | ||
required binary metadata; | ||
optional binary value; | ||
optional int64 typed_value; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
optional int64 typed_value; | |
// The exact semantics of this field are discussed in detail below, but this column stores the variant value when it is an integer. | |
optional int64 typed_value; |
|
||
Consider the following example: | ||
Statistics for `typed_value` columns can be used for file, row group, or page skipping when `value` is always null (missing). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Statistics for `typed_value` columns can be used for file, row group, or page skipping when `value` is always null (missing). | |
Min/Max statistics for `typed_value` columns may be used for row group and page skipping when `value` is always null (missing). The statistics may also be used if statistics can be used to infer the type of all non-null values are irrelevant to the min/max statistics (e.g. if all values are variant null (`00`) and a filter specifies equality to a integer). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think that we should make this more specific because anything further risks making assumptions about the semantics of the filter. The example on the dev list demonstrates this. This can't state that strings can be ignored for numeric filters because the filter may cast a string and result in a valid value.
### What changes were proposed in this pull request? It implements the variant rebuild functionality according to the current shredding spec in apache/parquet-format#461, and allows the Parquet reader will be able to read shredded variant data. ### Why are the changes needed? It gives Spark the basic ability to read shredded variant data. It can be improved in the future to read only requested fields. ### Does this PR introduce _any_ user-facing change? Yes, the Parquet reader will be able to read shredded variant data. ### How was this patch tested? Unit tests. ### Was this patch authored or co-authored using generative AI tooling? No. Closes #48851 from chenhao-db/rebuild_variant. Authored-by: Chenhao Li <chenhao.li@databricks.com> Signed-off-by: Wenchen Fan <wenchen@databricks.com>
### What changes were proposed in this pull request? The `variantShreddingSchema` method converts a human-readable schema for Variant to one that's a valid shredding schema. According to the shredding schema in apache/parquet-format#461, each shredded field in an object should be a required group - i.e. a non-nullable struct. This PR fixes the `variantShreddingSchema` to mark that struct as non-nullable. ### Why are the changes needed? If we use `variantShreddingSchema` to construct a schema for Parquet, the schema would be technically non-conformant with the spec by setting the group as optional. I don't think this should really matter to readers, but it would waste a bit of space in the Parquet file by adding an extra definition level. ### Does this PR introduce _any_ user-facing change? No, this code is not used yet. ### How was this patch tested? Added a test to do some minimal validation of the `variantShreddingSchema` function. ### Was this patch authored or co-authored using generative AI tooling? No. Closes #49118 from cashmand/SPARK-48898-nullability. Authored-by: cashmand <david.cashman@databricks.com> Signed-off-by: Wenchen Fan <wenchen@databricks.com>
|
||
At a high level, we replace the `value` field of the Variant Parquet group with one or more fields called `object`, `array`, `typed_value`, and `variant_value`. | ||
These represent a fixed schema suitable for constructing the full Variant value for each row. | ||
For example, the query `SELECT variant_get(event, '$.event_ts', 'timestamp') FROM tbl` only needs to load field `event_ts`, and if that column is shredded, it can be read by columnar projection without reading or deserializing the rest of the `event` Variant. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For example, the query `SELECT variant_get(event, '$.event_ts', 'timestamp') FROM tbl` only needs to load field `event_ts`, and if that column is shredded, it can be read by columnar projection without reading or deserializing the rest of the `event` Variant. | |
A partial projection occurs with a query like `SELECT variant_get(event, '$.event_ts', 'timestamp') FROM tbl`. In this case an engine only needs to load field `event_ts`, and if `event_ts` column is shredded, it can be read without reading or deserializing the rest of the `event` Variant. |
VariantShredding.md
Outdated
|
||
For an `object`, a null field means that the field does not exist in the reconstructed Variant object. | ||
All elements of an `array` must be non-null, since array elements cannote be missing. | ||
If both fields are non-null and either is not an object, the value is invalid. Readers must either fail or return the `typed_value`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If both fields are non-null and either is not an object, the value is invalid. Readers must either fail or return the `typed_value`. | |
If both fields are non-null and either is not an object, the value is invalid. Reader behavior is undefined[^2]. | |
[^2] The correct thing to do in this situation is rewrite the data file to fix the inconsistency. The correct way of rewriting the file can only be determined based on the bug that produced the inconsistency. Readers are encouraged to error this case to help identify writer bugs as soon as possible. Erroring might not be appropriate or possible given reader optimizations. If a reader does not error in this case, then favoring the shredded values would provide more consistent results (i.e. it works across both partial-projection and full reconstruction) but this does not make using the shredded values correct, simply more consistent. Similarly, any other type of "read side fix" is discouraged, but might be used by specific readers to help their consumers until data can be rewritten. |
@rdblue @julienledem I think this language might capture all of the concerns across (I think this wording applies to all potential incosistencies).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@rdblue and I also talked about this for a long time and I think I favor the current text. I feel like the additional text adds a bit of confusion around this.
A shredded reading of a field is always correct since a shredded reader will not be able to check an unshreddeed value for inconsistency, a reader using an un-shredded value when the shredded value is present is always incorrect.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@rdblue and I also talked about this for a long time and I think I favor the current text. I feel like the additional text adds a bit of confusion around this.
We also talked about it in the sync and didn't come to the conclusion. IIUC @rdblue wanted the error handling language to eliminate the possibility of trying to define an alternative behavior down the road by one specific reader. I don't think we should be mandating this in the spec but I do agree we should be clarifying this won't be relitigated.
A shredded reading of a field is always correct since a shredded reader will not be able to check an unshreddeed value for inconsistency, a reader using an un-shredded value when the shredded value is present is always incorrect.
I think semantics are important here. I tried to cover this by saying it is "consistent", i.e. it would always provided consistent results to the end user, which is a nice property to have and I agree most implementations should use it if they aren't going to error out.
"correct", I think, is a property of actually returning the the variant data without modification. Without understanding the bugs that introduced inconsistent shredding, I don't think it is possible say for sure the shredded values are correct.
The reason why I think it is important to say it is not "correct" is because in other instances (e.g. requiring no overlap between shredded/not shredded fields) to write out of spec files that would not face any issues (as an alternative I think it would OK to have overlapping values as long as they are consistent, then the real question becomes what happens if the values are not actually consistent, in this case I think it would really be very hard to understand what the correct results are).
As an analogy from another part of parquet (and a real world example I've encounted). Assume a schema like list<struct<required a int, required b int>>
we do not require readers to check that repetition levels and definition levels are consistent for columns a and b (and many don't). If they are not equal it is a bug, we can't really say which one, or if either is, "correct" without understanding the bug that introduced the inconsistency. A "consistent" result would be to always use the left most columns repetition and definition levels.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To maybe go a little bit deeper here, I think it is worth enumerating a matrix of potential operations and data inconsistencies (see below). The main purpose of these examples is to show that:
- Requiring using
typed_value
orfailure
pessimizes some cases or requires that different operations will return inconsistent result (making the value of trying to force consistency in some cases have less utility). - Preferring
typed_value
in the presence of some inconsistencies sseems like an arbitrary choice simply for the sake of not exposing bugs in the underlying data.
I think the way of rectifying this is either to add more cases that are considered "valid" (e.g. shredded object fields might overlap with fields in value as long as they are consistent) and/or leave the behavior undefined. My preference would simply be to do the latter.
I think there are three main operations that will be performed with Variant:
- Project as primitive non-strict - Return a field as a specific primitive value, if a field mismatches the type return null in its place.
- Project as primitive strict - Return a field specific primitive value if a field mismatches the type the query is failed.
- Project as variant - project a field as a variant type allowing for mixed types to be returned (this would also be projecting the top level variant to get the original value).
Primitive inconsistencies
Assume for all these cases a column is shredded as int32 as its typed value.
Inconsistency: Both typed_value and value are present for the same cell and the values are consistent with one another (e.g. all values in value
column are int32 and exactly equal to those in typed_value
Operation: Project field as int32 non-strict. In this case engines would prefer to always use typed_value and never need to read value
Operation: Project field as string not-strict. In this case only value
column would need to be read but would return consistent results as using typed_value
Operation: Project field as int32 as strict. If it is required that a field that can be projected always is (IIUC this isn't currently mandated by the spec), then this operation should fail on statististics as there would be a non-null
value
column. Otherwise this effectively requires the same logic projecting a variant described below.
Operation: Project field as string strict. This would fail based on stastistics from typed_value
.
Operation: Project field as variant. In this case both typed_value
and value
would need to be read and merged. When merging the reader could chose to always take the non-null value from typed_value
ignoring any present values in value
(assuming corresponding values are null). This would however lead to a strange state where after all values are merged, fewer cells from value
would be read then are present. An alternative would be check for consistency between repetition/definition levels, and realize they are both defined and still favor typed_value
, or check for consistency and continue reading as long as value
and typed_value
are consistent.
Inconsistency: Both type_value and value are present for the same cell and are inconsistent with one another (i.e. value
contains a string).
Operation: Project field as int32 non-strict. In this case engines would prefer to always use typed_value and never need to read value
. int32 value is returned over the string value
Operation: Project field as string not-strict. In this case only value
column would need to be read but would return inconsistent string value in the cells that conflict.
Operation: Project field as int32 as strict. Same as data inconsistency above.
Operation: Project field as string as strict. Same as data inconsistency above (fail based on statistics).
Operation: Project as variant. Same as data inconsistency above, but if the reader is checking for consistency between values to decide on what to do, as the current spec is written it would arbitrarily choose int32 value (valid alternatives seem like failure or returning the string
value.
Object inconsistencies
In this case assume there the object is shredded with a single child field "a" as int32. Objects overlap with the cases outlined above for consistency/inconsistency so they are not covered here, only somewhat novel cases are discussed below (merging is taken one step further because field overlap either must be detected or duplicate fields could be added when reconstructing the full variant).
Inconsistency: The value
object contains a non-object value (e.g. string) in the same cell that has a present value for the shredded column "a".
Operation: Project as as variant. In this case as I read the spec, it is valid for readers to return {a: <some int value>}
. Other valid options seem like (use the non-object value
or fail).
Inconsistency: The typed_value
object is marked as non-present but the value column
is present and contains object.
Operation: Project field as object only keeping fields "a" non-strict. In this case readers would ignore the value
column based on the spec and return null for the the inconsistency fields.
Operation: Project field only keep field "c" as variant. In this case the reader should be able to ignore typed_value
(it only contains an object "a" as a shredded column so it can be ignored) and return {"c": <some value>}
but that is not what the specification states.
### What changes were proposed in this pull request? The variantShreddingSchema method converts a human-readable schema for Variant to one that's a valid shredding schema. According to the shredding schema in apache/parquet-format#461, each shredded field in an object should be a required group - i.e. a non-nullable struct. This PR fixes the variantShreddingSchema to mark that struct as non-nullable. ### Why are the changes needed? If we use variantShreddingSchema to construct a schema for Parquet, the schema would be technically non-conformant with the spec by setting the group as optional. I don't think this should really matter to readers, but it would waste a bit of space in the Parquet file by adding an extra definition level. ### Does this PR introduce _any_ user-facing change? No, this code is not used yet. ### How was this patch tested? Added a test to do some minimal validation of the variantShreddingSchema function. ### Was this patch authored or co-authored using generative AI tooling? No. Closes #49151 from cashmand/SPARK-48898-nullability-again. Authored-by: cashmand <david.cashman@databricks.com> Signed-off-by: Wenchen Fan <wenchen@databricks.com>
On the other hand, shredding as a different logical type is not allowed. | ||
For example, the integer value 123 could not be shredded to a string `typed_value` column as the string "123", since that would lose type information. | ||
It would need to be written to the `variant_value` column. | ||
Each shredded field in the `typed_value` group is represented as a required group that contains optional `value` and `typed_value` fields. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Are there any cases where a field's name might not be a valid Parquet group name, making it impossible to shred? E.g. are there any restrictions on whitespace or non-printable characters in Parquet? Or any that should be discouraged for shredding, because readers are known to have trouble with them?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are no specific restrictions in the spec that I can recall, there are certainly engines that don't support some characters but shredded columns should hopefully never reach engine schemas since only the top level variant would ideally be displayed. For non-displayable characters general display and SQL projection functions would likely become cumbersome.
Some reference implementations might have issues with periods (.
) but from an API perspective it isn't clear to me if there should be a separate API anyways to interrogate if certain variants are shredded. This might be a design question to raise on the mailing list before beginning implementation.
### What changes were proposed in this pull request? The variantShreddingSchema method converts a human-readable schema for Variant to one that's a valid shredding schema. According to the shredding schema in apache/parquet-format#461, each shredded field in an object should be a required group - i.e. a non-nullable struct. This PR fixes the variantShreddingSchema to mark that struct as non-nullable. ### Why are the changes needed? If we use variantShreddingSchema to construct a schema for Parquet, the schema would be technically non-conformant with the spec by setting the group as optional. I don't think this should really matter to readers, but it would waste a bit of space in the Parquet file by adding an extra definition level. ### Does this PR introduce _any_ user-facing change? No, this code is not used yet. ### How was this patch tested? Added a test to do some minimal validation of the variantShreddingSchema function. ### Was this patch authored or co-authored using generative AI tooling? No. Closes apache#49151 from cashmand/SPARK-48898-nullability-again. Authored-by: cashmand <david.cashman@databricks.com> Signed-off-by: Wenchen Fan <wenchen@databricks.com>
} | ||
required group measurement (VARIANT) { | ||
required binary metadata; | ||
optional binary value; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @rdblue , I have thought of a specific scenario: suppose the json is a struct and has 4000 kvs, of which 100 keys intend to construct typed_value
. The remaining 3900 columns will not be shredded. In this case, they will all be stored in value
.
Is it possible for us to define the max number of keys contained in one binary value
, such as in this case, storing a maximum of 1000, so that v1, v2, v3, v4 can be generated to reduce scan IO.
Futher more, we can define the concept of group
and stipulate that some certain keys are divided into one binary value1
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@Zouxxyy could you expand on this use-case? Is there a specific query pattern for this type of division would improve performance for?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@emkornfield For the cols that we do not intend to shred, if we categorize them into four groups and happen to query fields within one of those groups, then our scan I/O can still be reduced by three-quarters.
Of course, I feel that the concept of group
will make shredding more complex (currently, it is concise and clear enough), I'm just looking to explore the possibilities and hear opinions.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks @Zouxxyy IIUC I think the suggestion is "partial shredding". Wouldn't this in most cases require consistent keys in the metadata portion (otherwise, one would need to do a full scan of the metadata column to determine all potential groups necessary, and it might turn out without consistent keys that multiple groups are needed for a single column)? If the keys are already all consistent, what are the trade-offs of partial shredding vs full shredding (i.e. all 4000 values are shredded)?
This goes back to my question on whether this is simply more theoretical at this point, or there are concrete use-cases where this suggestion would actually save a lot.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@Zouxxyy These would just be decisions made by the Shredder, the spec itself doesn't need a limit. For example you may make a "Shred Everything" shredder or you may have a "Shred only the most frequent field" shredder.
|
||
For an `object`, a null field means that the field does not exist in the reconstructed Variant object. | ||
All elements of an `array` must be non-null, since array elements cannote be missing. | ||
When `typed_value` is non-null, its value must always be used when reading. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
When `typed_value` is non-null, its value must always be used when reading. |
For an `object`, a null field means that the field does not exist in the reconstructed Variant object. | ||
All elements of an `array` must be non-null, since array elements cannote be missing. | ||
When `typed_value` is non-null, its value must always be used when reading. | ||
Writers must not produce data where both `value` and `typed_value` are non-null, unless the Variant value is an object. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Writers must not produce data where both `value` and `typed_value` are non-null, unless the Variant value is an object. | |
Writers must not produce data where both `value` and `typed_value` are non-null, unless the Variant value is an object. [^1] | |
[^1] If files are produced that do not conform to this requirement, reader behavior is undefined, and will not be defined. The recommended recourse for defined behavior is to rewrite the data correcting for any data inconsistencies. Readers may assume that data written conforms to the specification. Readers are free to optimize reading only `typed_value` and `value` to fulfill queries appropriately. Readers do not need to check for consistency due to performance overhead. If readers do check for consistency and find inconsistent data, readers should return an error. Based on these assumptions, correct behavior cannot be defined. Depending on the query readers might return logically inconsistent results, and therefore this is no behavior that is always correct. Hence favoring data from either `typed_value` or `value` simply masks deeper systemic issues. Therefore, changing implementations to favor one vs the other is discouraged. | |
|
||
``` | ||
optional group variant_name (VARIANT) { | ||
required binary metadata; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@chenhao-db @cashmand Hi, I noticed that currently Spark's Variant writes the value first and then the metadata. which is the opposite of shredding. Have we considered making adjustments to this?
spark's variant
optional group variant_name (VARIANT) {
required binary value;
required binary metadata;
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @Zouxxyy, in the Spark shredding PRs I've been working on, I put metadata first. I didn't see much benefit to changing the order in the existing non-shredded code, but I don't feel too strongly about it either way. The spec is pretty clear that readers should identify the appropriate columns based on field names, not field order, and I think things could become quite fragile if they did rely on field order.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @Zouxxyy, in the Spark shredding PRs I've been working on, I put metadata first. I didn't see much benefit to changing the order in the existing non-shredded code, but I don't feel too strongly about it either way. The spec is pretty clear that readers should identify the appropriate columns based on field names, not field order, and I think things could become quite fragile if they did rely on field order.
Thank you, but future users or developers may find it strange if the actual implementation differs from the specs. I think it's better to adhere to the specs as long as it doesn’t affect performance, before the official release of spark 4.0. If you don’t mind, I can work on this and raise a PR to spark, WDYT
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Okay, I think it's fine if you want to make a PR with the change.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Okay, I think it's fine if you want to make a PR with the change.
Oh, I checked again, and it seems that new VariantVal(byte[] value, byte[] metadata)
is already used everywhere in spark, this change comes at a significant cost. @cashmand @rdblue , do you think the inconsistency here with the specs will have a big impact in the future?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree that the discrepancy is a bit unfortunate. Later in the shredding doc it says The Parquet columns used to store variant metadata and values must be accessed by name, not by position
; maybe we should put it up front here.
Rationale for this change
Updating the Variant and shredding specs from a thorough review.
What changes are included in this PR?
Spec updates, mostly to the shredding spec to minimize it and make it clear. This also attempts to make the variant spec more consistent (for example, by using
value
in both).object
andarray
in favor of always usingtyped_value
required
to avoid unnecessary null casesmetadata
must be valid for all variant values without modificationDo these changes have PoC implementations?
No.