Skip to content

Commit

Permalink
fix: make PrimitiveLiteral and Literal not be Ord (#386)
Browse files Browse the repository at this point in the history
* make PrimitiveLiteral and Literal not be Ord

* refine Map

* fix name

* fix map test

* refine

---------

Co-authored-by: ZENOTME <[email protected]>
  • Loading branch information
ZENOTME and ZENOTME authored Jun 1, 2024
1 parent 4407634 commit 8068407
Show file tree
Hide file tree
Showing 7 changed files with 504 additions and 240 deletions.
87 changes: 28 additions & 59 deletions crates/iceberg/src/expr/visitors/inclusive_metrics_evaluator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@

use crate::expr::visitors::bound_predicate_visitor::{visit, BoundPredicateVisitor};
use crate::expr::{BoundPredicate, BoundReference};
use crate::spec::{DataFile, Datum, Literal, PrimitiveLiteral};
use crate::spec::{DataFile, Datum, PrimitiveLiteral};
use crate::{Error, ErrorKind};
use fnv::FnvHashSet;

Expand Down Expand Up @@ -63,11 +63,11 @@ impl<'a> InclusiveMetricsEvaluator<'a> {
self.data_file.value_counts.get(&field_id)
}

fn lower_bound(&self, field_id: i32) -> Option<&Literal> {
fn lower_bound(&self, field_id: i32) -> Option<&Datum> {
self.data_file.lower_bounds.get(&field_id)
}

fn upper_bound(&self, field_id: i32) -> Option<&Literal> {
fn upper_bound(&self, field_id: i32) -> Option<&Datum> {
self.data_file.upper_bounds.get(&field_id)
}

Expand Down Expand Up @@ -97,7 +97,7 @@ impl<'a> InclusiveMetricsEvaluator<'a> {
&mut self,
reference: &BoundReference,
datum: &Datum,
cmp_fn: fn(&PrimitiveLiteral, &PrimitiveLiteral) -> bool,
cmp_fn: fn(&Datum, &Datum) -> bool,
use_lower_bound: bool,
) -> crate::Result<bool> {
let field_id = reference.field().id;
Expand All @@ -119,14 +119,7 @@ impl<'a> InclusiveMetricsEvaluator<'a> {
};

if let Some(bound) = bound {
let Literal::Primitive(bound) = bound else {
return Err(Error::new(
ErrorKind::Unexpected,
"Inequality Predicates can only compare against a Primitive Literal",
));
};

if cmp_fn(bound, datum.literal()) {
if cmp_fn(bound, datum) {
return ROWS_MIGHT_MATCH;
}

Expand Down Expand Up @@ -265,33 +258,21 @@ impl BoundPredicateVisitor for InclusiveMetricsEvaluator<'_> {
}

if let Some(lower_bound) = self.lower_bound(field_id) {
let Literal::Primitive(lower_bound) = lower_bound else {
return Err(Error::new(
ErrorKind::Unexpected,
"Eq Predicate can only compare against a Primitive Literal",
));
};
if lower_bound.is_nan() {
// NaN indicates unreliable bounds.
// See the InclusiveMetricsEvaluator docs for more.
return ROWS_MIGHT_MATCH;
} else if lower_bound.gt(datum.literal()) {
} else if lower_bound.gt(datum) {
return ROWS_CANNOT_MATCH;
}
}

if let Some(upper_bound) = self.upper_bound(field_id) {
let Literal::Primitive(upper_bound) = upper_bound else {
return Err(Error::new(
ErrorKind::Unexpected,
"Eq Predicate can only compare against a Primitive Literal",
));
};
if upper_bound.is_nan() {
// NaN indicates unreliable bounds.
// See the InclusiveMetricsEvaluator docs for more.
return ROWS_MIGHT_MATCH;
} else if upper_bound.lt(datum.literal()) {
} else if upper_bound.lt(datum) {
return ROWS_CANNOT_MATCH;
}
}
Expand Down Expand Up @@ -331,7 +312,7 @@ impl BoundPredicateVisitor for InclusiveMetricsEvaluator<'_> {
};

if let Some(lower_bound) = self.lower_bound(field_id) {
let Literal::Primitive(PrimitiveLiteral::String(lower_bound)) = lower_bound else {
let PrimitiveLiteral::String(lower_bound) = lower_bound.literal() else {
return Err(Error::new(
ErrorKind::Unexpected,
"Cannot use StartsWith operator on non-string lower_bound value",
Expand All @@ -349,7 +330,7 @@ impl BoundPredicateVisitor for InclusiveMetricsEvaluator<'_> {
}

if let Some(upper_bound) = self.upper_bound(field_id) {
let Literal::Primitive(PrimitiveLiteral::String(upper_bound)) = upper_bound else {
let PrimitiveLiteral::String(upper_bound) = upper_bound.literal() else {
return Err(Error::new(
ErrorKind::Unexpected,
"Cannot use StartsWith operator on non-string upper_bound value",
Expand Down Expand Up @@ -395,7 +376,7 @@ impl BoundPredicateVisitor for InclusiveMetricsEvaluator<'_> {
return ROWS_MIGHT_MATCH;
};

let Literal::Primitive(PrimitiveLiteral::String(lower_bound_str)) = lower_bound else {
let PrimitiveLiteral::String(lower_bound_str) = lower_bound.literal() else {
return Err(Error::new(
ErrorKind::Unexpected,
"Cannot use NotStartsWith operator on non-string lower_bound value",
Expand All @@ -416,7 +397,7 @@ impl BoundPredicateVisitor for InclusiveMetricsEvaluator<'_> {
return ROWS_MIGHT_MATCH;
};

let Literal::Primitive(PrimitiveLiteral::String(upper_bound)) = upper_bound else {
let PrimitiveLiteral::String(upper_bound) = upper_bound.literal() else {
return Err(Error::new(
ErrorKind::Unexpected,
"Cannot use NotStartsWith operator on non-string upper_bound value",
Expand Down Expand Up @@ -456,36 +437,24 @@ impl BoundPredicateVisitor for InclusiveMetricsEvaluator<'_> {
}

if let Some(lower_bound) = self.lower_bound(field_id) {
let Literal::Primitive(lower_bound) = lower_bound else {
return Err(Error::new(
ErrorKind::Unexpected,
"Eq Predicate can only compare against a Primitive Literal",
));
};
if lower_bound.is_nan() {
// NaN indicates unreliable bounds. See the InclusiveMetricsEvaluator docs for more.
return ROWS_MIGHT_MATCH;
}

if !literals.iter().any(|datum| datum.literal().ge(lower_bound)) {
if !literals.iter().any(|datum| datum.ge(lower_bound)) {
// if all values are less than lower bound, rows cannot match.
return ROWS_CANNOT_MATCH;
}
}

if let Some(upper_bound) = self.upper_bound(field_id) {
let Literal::Primitive(upper_bound) = upper_bound else {
return Err(Error::new(
ErrorKind::Unexpected,
"Eq Predicate can only compare against a Primitive Literal",
));
};
if upper_bound.is_nan() {
// NaN indicates unreliable bounds. See the InclusiveMetricsEvaluator docs for more.
return ROWS_MIGHT_MATCH;
}

if !literals.iter().any(|datum| datum.literal().le(upper_bound)) {
if !literals.iter().any(|datum| datum.le(upper_bound)) {
// if all values are greater than upper bound, rows cannot match.
return ROWS_CANNOT_MATCH;
}
Expand Down Expand Up @@ -519,7 +488,7 @@ mod test {
UnaryExpression,
};
use crate::spec::{
DataContentType, DataFile, DataFileFormat, Datum, Literal, NestedField, PartitionField,
DataContentType, DataFile, DataFileFormat, Datum, NestedField, PartitionField,
PartitionSpec, PrimitiveType, Schema, Struct, Transform, Type,
};
use fnv::FnvHashSet;
Expand Down Expand Up @@ -2152,17 +2121,17 @@ mod test {
nan_value_counts: HashMap::from([(7, 50), (8, 10), (9, 0)]),

lower_bounds: HashMap::from([
(1, Literal::int(INT_MIN_VALUE)),
(11, Literal::float(f32::NAN)),
(12, Literal::double(f64::NAN)),
(14, Literal::string("")),
(1, Datum::int(INT_MIN_VALUE)),
(11, Datum::float(f32::NAN)),
(12, Datum::double(f64::NAN)),
(14, Datum::string("")),
]),

upper_bounds: HashMap::from([
(1, Literal::int(INT_MAX_VALUE)),
(11, Literal::float(f32::NAN)),
(12, Literal::double(f64::NAN)),
(14, Literal::string("房东整租霍营小区二层两居室")),
(1, Datum::int(INT_MAX_VALUE)),
(11, Datum::float(f32::NAN)),
(12, Datum::double(f64::NAN)),
(14, Datum::string("房东整租霍营小区二层两居室")),
]),

column_sizes: Default::default(),
Expand All @@ -2187,9 +2156,9 @@ mod test {

nan_value_counts: HashMap::default(),

lower_bounds: HashMap::from([(3, Literal::string("aa"))]),
lower_bounds: HashMap::from([(3, Datum::string("aa"))]),

upper_bounds: HashMap::from([(3, Literal::string("dC"))]),
upper_bounds: HashMap::from([(3, Datum::string("dC"))]),

column_sizes: Default::default(),
key_metadata: vec![],
Expand All @@ -2214,9 +2183,9 @@ mod test {

nan_value_counts: HashMap::default(),

lower_bounds: HashMap::from([(3, Literal::string("1str1"))]),
lower_bounds: HashMap::from([(3, Datum::string("1str1"))]),

upper_bounds: HashMap::from([(3, Literal::string("3str3"))]),
upper_bounds: HashMap::from([(3, Datum::string("3str3"))]),

column_sizes: Default::default(),
key_metadata: vec![],
Expand All @@ -2241,9 +2210,9 @@ mod test {

nan_value_counts: HashMap::default(),

lower_bounds: HashMap::from([(3, Literal::string("abc"))]),
lower_bounds: HashMap::from([(3, Datum::string("abc"))]),

upper_bounds: HashMap::from([(3, Literal::string("イロハニホヘト"))]),
upper_bounds: HashMap::from([(3, Datum::string("イロハニホヘト"))]),

column_sizes: Default::default(),
key_metadata: vec![],
Expand Down
11 changes: 10 additions & 1 deletion crates/iceberg/src/spec/datatypes.rs
Original file line number Diff line number Diff line change
Expand Up @@ -118,8 +118,17 @@ impl Type {
matches!(self, Type::Struct(_) | Type::List(_) | Type::Map(_))
}

/// Convert Type to reference of PrimitiveType
pub fn as_primitive_type(&self) -> Option<&PrimitiveType> {
if let Type::Primitive(primitive_type) = self {
Some(primitive_type)
} else {
None
}
}

/// Convert Type to StructType
pub fn as_struct_type(self) -> Option<StructType> {
pub fn to_struct_type(self) -> Option<StructType> {
if let Type::Struct(struct_type) = self {
Some(struct_type)
} else {
Expand Down
Loading

0 comments on commit 8068407

Please sign in to comment.