From b626a829d57a52642d00e4310c2e78c0772af61f Mon Sep 17 00:00:00 2001 From: duonganhthu43 Date: Tue, 16 Apr 2024 11:54:22 +0700 Subject: [PATCH] Fix/clickhouse sink issue with collapsing merge tree (#2477) * fix/clickhouse with Collapsing Merge Tree * revert back to default MergeTree * fix: clippy fmt --- dozer-ingestion/aerospike/src/connector.rs | 5 + dozer-ingestion/mysql/src/conversion.rs | 1 + dozer-ingestion/tests/test_suite/basic.rs | 3 + .../connectors/object_store/arrow.rs | 13 + .../tests/test_suite/connectors/sql.rs | 2 + dozer-ingestion/webhook/src/util.rs | 5 + dozer-sink-aerospike/src/aerospike.rs | 18 + dozer-sink-aerospike/src/lib.rs | 1 + dozer-sink-clickhouse/src/ddl.rs | 19 +- dozer-sink-clickhouse/src/schema.rs | 1 + dozer-sink-clickhouse/src/sink.rs | 27 +- dozer-sink-clickhouse/src/types.rs | 2 +- dozer-sink-oracle/src/lib.rs | 1 + dozer-sql/expression/src/cast.rs | 24 + dozer-sql/expression/src/comparison/mod.rs | 254 +++++++++ dozer-sql/expression/src/execution.rs | 8 + dozer-sql/expression/src/logical.rs | 7 + dozer-sql/expression/src/mathematical/mod.rs | 534 ++++++++++++++++++ dozer-sql/expression/src/python_udf.rs | 1 + dozer-sql/expression/src/scalar/number.rs | 3 + dozer-sql/expression/src/scalar/string.rs | 4 + dozer-sql/src/aggregation/avg.rs | 8 + dozer-sql/src/aggregation/count.rs | 1 + dozer-sql/src/aggregation/max_append_only.rs | 11 + dozer-sql/src/aggregation/min_append_only.rs | 11 + dozer-sql/src/aggregation/sum.rs | 16 + dozer-tests/src/sql_tests/helper/mapper.rs | 1 + dozer-types/src/arrow_types/to_arrow.rs | 1 + dozer-types/src/grpc_types.rs | 4 + dozer-types/src/helper.rs | 17 + dozer-types/src/json_types.rs | 1 + dozer-types/src/types/field.rs | 31 + 32 files changed, 1021 insertions(+), 14 deletions(-) diff --git a/dozer-ingestion/aerospike/src/connector.rs b/dozer-ingestion/aerospike/src/connector.rs index 663c5fdbcb..c87fdef157 100644 --- a/dozer-ingestion/aerospike/src/connector.rs +++ b/dozer-ingestion/aerospike/src/connector.rs @@ -807,6 +807,11 @@ pub(crate) fn map_value_to_field( AerospikeConnectorError::ParsingIntFailed })?)) } + FieldType::Int8 => { + check_type("int8")?; + let string = value.as_str().ok_or_else(unsupported_type)?; + Ok(Field::Int8(string.parse()?)) + } FieldType::U128 => { check_type("str")?; let string = value.as_str().ok_or_else(unsupported_type)?; diff --git a/dozer-ingestion/mysql/src/conversion.rs b/dozer-ingestion/mysql/src/conversion.rs index 51ea0348e2..98ef8c3415 100644 --- a/dozer-ingestion/mysql/src/conversion.rs +++ b/dozer-ingestion/mysql/src/conversion.rs @@ -193,6 +193,7 @@ impl<'a> IntoField<'a> for Value { FieldType::UInt => Field::UInt(from_value_opt::(value)?), FieldType::U128 => Field::U128(from_value_opt::(value)?), FieldType::Int => Field::Int(from_value_opt::(value)?), + FieldType::Int8 => Field::Int8(from_value_opt::(value)?), FieldType::I128 => Field::I128(from_value_opt::(value)?), FieldType::Float => Field::Float(from_value_opt::(value)?.into()), FieldType::Boolean => Field::Boolean(from_value_opt::(value)?), diff --git a/dozer-ingestion/tests/test_suite/basic.rs b/dozer-ingestion/tests/test_suite/basic.rs index c86e975421..2f1e30f883 100644 --- a/dozer-ingestion/tests/test_suite/basic.rs +++ b/dozer-ingestion/tests/test_suite/basic.rs @@ -293,6 +293,9 @@ fn assert_record_matches_schema(record: &Record, schema: &Schema, only_match_pk: FieldType::Int => { assert!(value.as_int().is_some()) } + FieldType::Int8 => { + assert!(value.as_int().is_some()) + } FieldType::I128 => { assert!(value.as_i128().is_some()) } diff --git a/dozer-ingestion/tests/test_suite/connectors/object_store/arrow.rs b/dozer-ingestion/tests/test_suite/connectors/object_store/arrow.rs index c3bc3ec4f1..5376a15df7 100644 --- a/dozer-ingestion/tests/test_suite/connectors/object_store/arrow.rs +++ b/dozer-ingestion/tests/test_suite/connectors/object_store/arrow.rs @@ -278,6 +278,7 @@ fn field_type_to_arrow(field_type: FieldType) -> Option Some(arrow::datatypes::DataType::UInt64), FieldType::U128 => None, FieldType::Int => Some(arrow::datatypes::DataType::Int64), + FieldType::Int8 => Some(arrow::datatypes::DataType::Int64), FieldType::I128 => None, FieldType::Float => Some(arrow::datatypes::DataType::Float64), FieldType::Boolean => Some(arrow::datatypes::DataType::Boolean), @@ -349,6 +350,18 @@ fn fields_to_arrow<'a, F: IntoIterator>( } Arc::new(builder.finish()) } + FieldType::Int8 => { + let mut builder = arrow::array::Int64Array::builder(count); + for field in fields { + match field { + Field::Int(value) => builder.append_value(*value), + Field::Int8(value) => builder.append_value(*value as i64), + Field::Null => builder.append_null(), + _ => panic!("Unexpected field type"), + } + } + Arc::new(builder.finish()) + } FieldType::I128 => panic!("Unexpected field type"), FieldType::Float => { let mut builder = arrow::array::Float64Array::builder(count); diff --git a/dozer-ingestion/tests/test_suite/connectors/sql.rs b/dozer-ingestion/tests/test_suite/connectors/sql.rs index 6268456915..3608f1214c 100644 --- a/dozer-ingestion/tests/test_suite/connectors/sql.rs +++ b/dozer-ingestion/tests/test_suite/connectors/sql.rs @@ -158,6 +158,7 @@ fn field_type_to_sql(field_type: FieldType) -> Option { FieldType::UInt => None, FieldType::U128 => None, FieldType::Int => Some("INT8".to_string()), + FieldType::Int8 => Some("INT8".to_string()), FieldType::I128 => None, FieldType::Float => Some("FLOAT8".to_string()), FieldType::Boolean => Some("BOOLEAN".to_string()), @@ -229,6 +230,7 @@ fn field_to_sql(field: &Field) -> String { Field::UInt(i) => i.to_string(), Field::U128(i) => i.to_string(), Field::Int(i) => i.to_string(), + Field::Int8(i) => i.to_string(), Field::I128(i) => i.to_string(), Field::Float(f) => f.to_string(), Field::Boolean(b) => b.to_string(), diff --git a/dozer-ingestion/webhook/src/util.rs b/dozer-ingestion/webhook/src/util.rs index f3ddd6e461..333a30c09a 100644 --- a/dozer-ingestion/webhook/src/util.rs +++ b/dozer-ingestion/webhook/src/util.rs @@ -52,6 +52,11 @@ pub fn map_record( let field = Field::Int(i64_value); values.push(field); } + FieldType::Int8 => { + let i8_value: i8 = serde_json::from_value(value.clone())?; + let field = Field::Int8(i8_value); + values.push(field); + } FieldType::Float => { let float_value: f64 = serde_json::from_value(value.clone())?; let field = Field::Float(OrderedFloat(float_value)); diff --git a/dozer-sink-aerospike/src/aerospike.rs b/dozer-sink-aerospike/src/aerospike.rs index 5deb836801..d6d59d9a1e 100644 --- a/dozer-sink-aerospike/src/aerospike.rs +++ b/dozer-sink-aerospike/src/aerospike.rs @@ -477,6 +477,9 @@ unsafe fn init_key_single( Field::Int(v) => { as_key_init_int64(key, namespace.as_ptr(), set.as_ptr(), *v); } + Field::Int8(v) => { + as_key_init_int64(key, namespace.as_ptr(), set.as_ptr(), (*v).into()); + } Field::U128(v) => set_str_key(key, namespace, set, v.to_string(), allocated_strings), Field::I128(v) => set_str_key(key, namespace, set, v.to_string(), allocated_strings), Field::Decimal(v) => set_str_key(key, namespace, set, v.to_string(), allocated_strings), @@ -554,6 +557,13 @@ pub(crate) unsafe fn new_record_map( Field::Int(v) => { as_orderedmap_set(map, key, check_alloc(as_integer_new(*v)) as *const as_val); } + Field::Int8(v) => { + as_orderedmap_set( + map, + key, + check_alloc(as_integer_new((*v).into())) as *const as_val, + ); + } Field::I128(v) => { map_set_str(map, key, v, allocated_strings); } @@ -662,6 +672,9 @@ pub(crate) unsafe fn init_batch_write_operations( Field::Int(v) => { as_operations_add_write_int64(ops, name, *v); } + Field::Int8(v) => { + as_operations_add_write_int64(ops, name, (*v).into()); + } Field::I128(v) => { set_operation_str(ops, name, v.to_string(), allocated_strings); } @@ -806,6 +819,11 @@ fn parse_val( Some(Field::Int(v.value)) }) } + dozer_types::types::FieldType::Int8 => { + map(val, as_val_type_e_AS_INTEGER, |v: &as_integer| { + Some(Field::Int8(v.value as i8)) + }) + } dozer_types::types::FieldType::I128 => { map(val, as_val_type_e_AS_STRING, |v: &as_string| { Some(Field::I128(unsafe { diff --git a/dozer-sink-aerospike/src/lib.rs b/dozer-sink-aerospike/src/lib.rs index c28937932b..1a3c20675f 100644 --- a/dozer-sink-aerospike/src/lib.rs +++ b/dozer-sink-aerospike/src/lib.rs @@ -146,6 +146,7 @@ impl SinkFactory for AerospikeSinkFactory { dozer_types::types::FieldType::UInt | dozer_types::types::FieldType::U128 | dozer_types::types::FieldType::Int + | dozer_types::types::FieldType::Int8 | dozer_types::types::FieldType::I128 | dozer_types::types::FieldType::String | dozer_types::types::FieldType::Text diff --git a/dozer-sink-clickhouse/src/ddl.rs b/dozer-sink-clickhouse/src/ddl.rs index 0610b1e41a..aa4da3f1bf 100644 --- a/dozer-sink-clickhouse/src/ddl.rs +++ b/dozer-sink-clickhouse/src/ddl.rs @@ -10,6 +10,15 @@ pub fn get_create_table_query( fields: &[FieldDefinition], table_options: Option, ) -> String { + let engine = table_options + .as_ref() + .and_then(|c| c.engine.clone()) + .unwrap_or_else(|| DEFAULT_TABLE_ENGINE.to_string()); + let engine_name = if engine == "CollapsingMergeTree" { + "CollapsingMergeTree(sign)".to_string() + } else { + engine.to_owned() + }; let mut parts = fields .iter() .map(|field| { @@ -17,11 +26,9 @@ pub fn get_create_table_query( format!("{} {}", field.name, typ) }) .collect::>(); - - let engine = table_options - .as_ref() - .and_then(|c| c.engine.clone()) - .unwrap_or_else(|| DEFAULT_TABLE_ENGINE.to_string()); + if engine == "CollapsingMergeTree" { + parts.push("sign Int8".to_string()); + } parts.push( table_options @@ -63,7 +70,7 @@ pub fn get_create_table_query( "CREATE TABLE IF NOT EXISTS {table_name} {cluster} ( {query} ) - ENGINE = {engine} + ENGINE = {engine_name} {order_by} {partition_by} {sample_by} diff --git a/dozer-sink-clickhouse/src/schema.rs b/dozer-sink-clickhouse/src/schema.rs index 2c45cc11b4..14c21f0a30 100644 --- a/dozer-sink-clickhouse/src/schema.rs +++ b/dozer-sink-clickhouse/src/schema.rs @@ -150,6 +150,7 @@ pub fn map_field_to_type(field: &FieldDefinition) -> String { FieldType::UInt => "UInt64", FieldType::U128 => "UInt128", FieldType::Int => "Int64", + FieldType::Int8 => "Int8", FieldType::I128 => "Int128", FieldType::Float => "Float64", FieldType::Boolean => "Boolean", diff --git a/dozer-sink-clickhouse/src/sink.rs b/dozer-sink-clickhouse/src/sink.rs index 4b0c70b4bb..06dbd8e1a2 100644 --- a/dozer-sink-clickhouse/src/sink.rs +++ b/dozer-sink-clickhouse/src/sink.rs @@ -16,7 +16,7 @@ use crate::metadata::{ }; use crate::schema::{ClickhouseSchema, ClickhouseTable}; use dozer_types::tonic::async_trait; -use dozer_types::types::{Field, Operation, Schema, TableOperation}; +use dozer_types::types::{Field, FieldDefinition, Operation, Schema, TableOperation}; use std::collections::HashMap; use std::fmt::Debug; use std::sync::Arc; @@ -113,7 +113,6 @@ impl SinkFactory for ClickhouseSinkFactory { let table = ClickhouseSchema::get_clickhouse_table(client.clone(), &self.config).await?; ClickhouseSchema::compare_with_dozer_schema(client.clone(), &schema, &table).await?; - let sink = ClickhouseSink::new( client, self.config.clone(), @@ -155,6 +154,19 @@ impl ClickhouseSink { runtime: Arc, table: ClickhouseTable, ) -> Self { + let mut schema = schema.clone(); + + if table.engine == "CollapsingMergeTree" && !schema.fields.is_empty() { + // get source from any field in schema + let source = schema.fields[0].source.clone(); + schema.fields.push(FieldDefinition { + name: "sign".to_string(), + typ: dozer_types::types::FieldType::Int8, + nullable: false, + description: None, + source, + }); + } Self { client, runtime, @@ -249,8 +261,7 @@ impl Sink for ClickhouseSink { Operation::Insert { new } => { if self.table.engine == "CollapsingMergeTree" { let mut values = new.values; - values.push(Field::Int(1)); - + values.push(Field::Int8(1)); self.insert_values(&values)?; } else { self.insert_values(&new.values)?; @@ -273,17 +284,19 @@ impl Sink for ClickhouseSink { return Err(BoxedError::from(ClickhouseSinkError::UnsupportedOperation)); } let mut values = old.values; - values.push(Field::Int(-1)); + values.push(Field::Int8(-1)); self.insert_values(&values)?; let mut values = new.values; - values.push(Field::Int(1)); + values.push(Field::Int8(1)); self.insert_values(&values)?; } Operation::BatchInsert { new } => { for record in new { let mut values = record.values; - values.push(Field::Int(1)); + if self.table.engine == "CollapsingMergeTree" { + values.push(Field::Int8(1)); + } self.insert_values(&values)?; } self.commit_batch()?; diff --git a/dozer-sink-clickhouse/src/types.rs b/dozer-sink-clickhouse/src/types.rs index 6b13a0055a..1994b6b3c6 100644 --- a/dozer-sink-clickhouse/src/types.rs +++ b/dozer-sink-clickhouse/src/types.rs @@ -229,6 +229,7 @@ fn add_last_column_to_block( FieldType::UInt => add_last_column.call(trivial_mapper!(Field::UInt)), FieldType::U128 => add_last_column.call(trivial_mapper!(Field::U128)), FieldType::Int => add_last_column.call(trivial_mapper!(Field::Int)), + FieldType::Int8 => add_last_column.call(trivial_mapper!(Field::Int8)), FieldType::I128 => add_last_column.call(trivial_mapper!(Field::I128)), FieldType::Boolean => add_last_column.call(trivial_mapper!(Field::Boolean)), FieldType::Float => add_last_column.call(|field| match field { @@ -276,7 +277,6 @@ pub async fn insert_multi( query_id: Option, ) -> Result<(), QueryError> { let mut block = Block::::new(); - for field in fields.iter().rev() { block = add_last_column_to_block(block, &field.name, &mut rows, field.typ, field.nullable)?; } diff --git a/dozer-sink-oracle/src/lib.rs b/dozer-sink-oracle/src/lib.rs index 71e9594476..56e288a765 100644 --- a/dozer-sink-oracle/src/lib.rs +++ b/dozer-sink-oracle/src/lib.rs @@ -280,6 +280,7 @@ impl OracleSinkFactory { FieldType::UInt => "NUMBER(20)", FieldType::U128 => unimplemented!(), FieldType::Int => "NUMBER(20)", + FieldType::Int8 => unimplemented!(), FieldType::I128 => unimplemented!(), // Should this be BINARY_DOUBLE? FieldType::Float => "NUMBER", diff --git a/dozer-sql/expression/src/cast.rs b/dozer-sql/expression/src/cast.rs index 47e966dfed..7c84720676 100644 --- a/dozer-sql/expression/src/cast.rs +++ b/dozer-sql/expression/src/cast.rs @@ -21,6 +21,7 @@ impl Display for CastOperatorType { FieldType::UInt => f.write_str("CAST AS UINT"), FieldType::U128 => f.write_str("CAST AS U128"), FieldType::Int => f.write_str("CAST AS INT"), + FieldType::Int8 => f.write_str("CAST AS INT8"), FieldType::I128 => f.write_str("CAST AS I128"), FieldType::Float => f.write_str("CAST AS FLOAT"), FieldType::Boolean => f.write_str("CAST AS BOOLEAN"), @@ -79,6 +80,19 @@ impl CastOperatorType { FieldType::Int => ( vec![ FieldType::Int, + FieldType::Int8, + FieldType::String, + FieldType::UInt, + FieldType::I128, + FieldType::U128, + FieldType::Json, + ], + FieldType::Int, + ), + FieldType::Int8 => ( + vec![ + FieldType::Int, + FieldType::Int8, FieldType::String, FieldType::UInt, FieldType::I128, @@ -252,6 +266,16 @@ pub fn cast_field(input: &Field, output_type: FieldType) -> Result }) } } + FieldType::Int8 => { + if let Some(value) = input.to_int8() { + Ok(Field::Int8(value)) + } else { + Err(Error::InvalidCast { + from: input.clone(), + to: FieldType::Int, + }) + } + } FieldType::I128 => { if let Some(value) = input.to_i128() { Ok(Field::I128(value)) diff --git a/dozer-sql/expression/src/comparison/mod.rs b/dozer-sql/expression/src/comparison/mod.rs index 68ac8015ca..644815288d 100644 --- a/dozer-sql/expression/src/comparison/mod.rs +++ b/dozer-sql/expression/src/comparison/mod.rs @@ -65,6 +65,16 @@ macro_rules! define_comparison { })?; Ok(Field::Boolean($function(left_v, right_v_b))) } + Field::Int8(right_v) => { + let right_v_b = + bool::from_str(right_v.to_string().as_str()).map_err(|_| { + PipelineError::UnableToCast( + format!("{}", right_v), + "Bool".to_string(), + ) + })?; + Ok(Field::Boolean($function(left_v, right_v_b))) + } // left: Bool, right: I128 Field::I128(right_v) => { let right_v_b = @@ -109,6 +119,8 @@ macro_rules! define_comparison { Field::Int(left_v) => match right_p { // left: Int, right: Int Field::Int(right_v) => Ok(Field::Boolean($function(left_v, right_v))), + Field::Int8(right_v) => Ok(Field::Boolean($function(left_v, right_v as i64))), + // left: Int, right: I128 Field::I128(right_v) => Ok(Field::Boolean($function(left_v as i128, right_v))), // left: Int, right: UInt @@ -154,9 +166,66 @@ macro_rules! define_comparison { $op.to_string(), )), }, + Field::Int8(left_v) => match right_p { + // left: Int, right: Int + Field::Int(right_v) => Ok(Field::Boolean($function(left_v as i64, right_v))), + Field::Int8(right_v) => { + Ok(Field::Boolean($function(left_v as i64, right_v as i64))) + } + + // left: Int, right: I128 + Field::I128(right_v) => Ok(Field::Boolean($function(left_v as i128, right_v))), + // left: Int, right: UInt + Field::UInt(right_v) => { + Ok(Field::Boolean($function(left_v as i64, right_v as i64))) + } + // left: Int, right: U128 + Field::U128(right_v) => { + Ok(Field::Boolean($function(left_v as i64, right_v as i64))) + } + // left: Int, right: Float + Field::Float(right_v) => Ok(Field::Boolean($function(left_v as f64, *right_v))), + // left: Int, right: Decimal + Field::Decimal(right_v) => { + let left_v_d = + Decimal::from_i64(left_v as i64).ok_or(PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ))?; + Ok(Field::Boolean($function(left_v_d, right_v))) + } + // left: Int, right: String or Text + Field::String(right_v) | Field::Text(right_v) => { + let right_v_b = i64::from_str(right_v.as_str()).map_err(|_| { + PipelineError::UnableToCast(format!("{}", right_v), "Int".to_string()) + })?; + Ok(Field::Boolean($function(left_v as i64, right_v_b))) + } + // left: Int, right: Duration + Field::Duration(right_v) => { + let left_v_b = DozerDuration( + Duration::from_nanos(left_v as u64), + TimeUnit::Nanoseconds, + ); + Ok(Field::Boolean($function(left_v_b, right_v))) + } + // left: Int, right: Null + Field::Null => Ok(Field::Null), + Field::Boolean(_) + | Field::Binary(_) + | Field::Timestamp(_) + | Field::Date(_) + | Field::Json(_) + | Field::Point(_) => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }, Field::I128(left_v) => match right_p { // left: I128, right: Int Field::Int(right_v) => Ok(Field::Boolean($function(left_v, right_v as i128))), + Field::Int8(right_v) => Ok(Field::Boolean($function(left_v, right_v as i128))), // left: I128, right: I128 Field::I128(right_v) => Ok(Field::Boolean($function(left_v, right_v))), // left: I128, right: UInt @@ -205,6 +274,9 @@ macro_rules! define_comparison { Field::UInt(left_v) => match right_p { // left: UInt, right: Int Field::Int(right_v) => Ok(Field::Boolean($function(left_v as i64, right_v))), + Field::Int8(right_v) => { + Ok(Field::Boolean($function(left_v as i64, right_v as i64))) + } // left: UInt, right: I128 Field::I128(right_v) => Ok(Field::Boolean($function(left_v as i128, right_v))), // left: UInt, right: UInt @@ -253,6 +325,9 @@ macro_rules! define_comparison { Field::Int(right_v) => { Ok(Field::Boolean($function(left_v as i128, right_v as i128))) } + Field::Int8(right_v) => { + Ok(Field::Boolean($function(left_v as i128, right_v as i128))) + } // left: U128, right: I128 Field::I128(right_v) => Ok(Field::Boolean($function(left_v as i128, right_v))), // left: U128, right: UInt @@ -303,10 +378,13 @@ macro_rules! define_comparison { Field::Float(right_v) => Ok(Field::Boolean($function(left_v, right_v))), // left: Float, right: UInt Field::UInt(right_v) => Ok(Field::Boolean($function(*left_v, right_v as f64))), + // left: Float, right: U128 Field::U128(right_v) => Ok(Field::Boolean($function(*left_v, right_v as f64))), // left: Float, right: Int Field::Int(right_v) => Ok(Field::Boolean($function(*left_v, right_v as f64))), + Field::Int8(right_v) => Ok(Field::Boolean($function(*left_v, right_v as f64))), + // left: Float, right: I128 Field::I128(right_v) => Ok(Field::Boolean($function(*left_v, right_v as f64))), // left: Float, right: Decimal @@ -358,6 +436,15 @@ macro_rules! define_comparison { ))?; Ok(Field::Boolean($function(left_v, right_v_d))) } + Field::Int8(right_v) => { + let right_v_d = Decimal::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "Decimal".to_string(), + ), + )?; + Ok(Field::Boolean($function(left_v, right_v_d))) + } // left: Decimal, right: I128 Field::I128(right_v) => { let right_v_d = @@ -433,6 +520,12 @@ macro_rules! define_comparison { })?; Ok(Field::Boolean($function(left_val, right_v))) } + Field::Int8(right_v) => { + let left_val = i64::from_str(left_v).map_err(|_| { + PipelineError::UnableToCast(format!("{}", left_v), "Int8".to_string()) + })?; + Ok(Field::Boolean($function(left_val, right_v as i64))) + } Field::I128(right_v) => { let left_val = i128::from_str(left_v).map_err(|_| { PipelineError::UnableToCast(format!("{}", left_v), "I128".to_string()) @@ -515,6 +608,7 @@ macro_rules! define_comparison { Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -545,6 +639,7 @@ macro_rules! define_comparison { Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -571,6 +666,7 @@ macro_rules! define_comparison { Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -601,6 +697,7 @@ macro_rules! define_comparison { Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -667,6 +764,13 @@ pub fn evaluate_lt( })?; Ok(Field::Boolean(!left_v & right_v_b)) } + + Field::Int8(right_v) => { + let right_v_b = bool::from_str(right_v.to_string().as_str()).map_err(|_| { + PipelineError::UnableToCast(format!("{}", right_v), "Bool".to_string()) + })?; + Ok(Field::Boolean(!left_v & right_v_b)) + } // left: Bool, right: I128 Field::I128(right_v) => { let right_v_b = bool::from_str(right_v.to_string().as_str()).map_err(|_| { @@ -699,6 +803,8 @@ pub fn evaluate_lt( Field::Int(left_v) => match right_p { // left: Int, right: Int Field::Int(right_v) => Ok(Field::Boolean(left_v < right_v)), + Field::Int8(right_v) => Ok(Field::Boolean(left_v < (right_v as i64))), + // left: Int, right: I128 Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) < right_v)), // left: Int, right: UInt @@ -741,13 +847,62 @@ pub fn evaluate_lt( "<".to_string(), )), }, + Field::Int8(left_v) => match right_p { + // left: Int, right: Int + Field::Int(right_v) => Ok(Field::Boolean((left_v as i64) < right_v)), + Field::Int8(right_v) => Ok(Field::Boolean(left_v < right_v)), + + // left: Int, right: I128 + Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) < right_v)), + // left: Int, right: UInt + Field::UInt(right_v) => Ok(Field::Boolean((left_v as i64) < (right_v as i64))), + // left: Int, right: U128 + Field::U128(right_v) => Ok(Field::Boolean((left_v as i64) < (right_v as i64))), + // left: Int, right: Float + Field::Float(right_v) => Ok(Field::Boolean((left_v as f64) < *right_v)), + // left: Int, right: Decimal + Field::Decimal(right_v) => { + let left_v_d = Decimal::from_i64(left_v as i64).ok_or( + PipelineError::UnableToCast(format!("{}", left_v), "Decimal".to_string()), + )?; + Ok(Field::Boolean(left_v_d < right_v)) + } + // left: Int, right: String or Text + Field::String(right_v) | Field::Text(right_v) => { + let right_v_b = i8::from_str(right_v.as_str()).map_err(|_| { + PipelineError::UnableToCast(right_v.to_string(), "Int".to_string()) + })?; + Ok(Field::Boolean(left_v < right_v_b)) + } + // left: Int, right: Duration + Field::Duration(right_v) => { + let left_v_b = + DozerDuration(Duration::from_nanos(left_v as u64), TimeUnit::Nanoseconds); + Ok(Field::Boolean(left_v_b < right_v)) + } + // left: Int, right: Null + Field::Null => Ok(Field::Null), + Field::Boolean(_) + | Field::Binary(_) + | Field::Timestamp(_) + | Field::Date(_) + | Field::Json(_) + | Field::Point(_) => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + "<".to_string(), + )), + }, Field::I128(left_v) => match right_p { // left: I128, right: Int Field::Int(right_v) => Ok(Field::Boolean(left_v < (right_v as i128))), + Field::Int8(right_v) => Ok(Field::Boolean(left_v < (right_v as i128))), + // left: I128, right: I128 Field::I128(right_v) => Ok(Field::Boolean(left_v < right_v)), // left: I128, right: UInt Field::UInt(right_v) => Ok(Field::Boolean(left_v < (right_v as i128))), + // left: I128, right: U128 Field::U128(right_v) => Ok(Field::Boolean(left_v < (right_v as i128))), // left: I128, right: Float @@ -789,6 +944,8 @@ pub fn evaluate_lt( Field::UInt(left_v) => match right_p { // left: UInt, right: Int Field::Int(right_v) => Ok(Field::Boolean((left_v as i64) < right_v)), + Field::Int8(right_v) => Ok(Field::Boolean((left_v as i64) < (right_v as i64))), + // left: UInt, right: I128 Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) < right_v)), // left: UInt, right: UInt @@ -832,6 +989,8 @@ pub fn evaluate_lt( Field::U128(left_v) => match right_p { // left: U128, right: Int Field::Int(right_v) => Ok(Field::Boolean((left_v as i128) < (right_v as i128))), + Field::Int8(right_v) => Ok(Field::Boolean((left_v as i128) < (right_v as i128))), + // left: U128, right: I128 Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) < right_v)), // left: U128, right: UInt @@ -882,6 +1041,8 @@ pub fn evaluate_lt( Field::U128(right_v) => Ok(Field::Boolean(*left_v < (right_v as f64))), // left: Float, right: Int Field::Int(right_v) => Ok(Field::Boolean(*left_v < (right_v as f64))), + Field::Int8(right_v) => Ok(Field::Boolean(*left_v < (right_v as f64))), + // left: Float, right: I128 Field::I128(right_v) => Ok(Field::Boolean(*left_v < (right_v as f64))), // left: Float, right: Decimal @@ -929,6 +1090,12 @@ pub fn evaluate_lt( )?; Ok(Field::Boolean(left_v < right_v_d)) } + Field::Int8(right_v) => { + let right_v_d = Decimal::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast(format!("{}", right_v), "Decimal".to_string()), + )?; + Ok(Field::Boolean(left_v < right_v_d)) + } // left: Decimal, right: I128 Field::I128(right_v) => { let right_v_d = Decimal::from_i128(right_v).ok_or( @@ -996,6 +1163,12 @@ pub fn evaluate_lt( })?; Ok(Field::Boolean(left_val < right_v)) } + Field::Int8(right_v) => { + let left_val = i64::from_str(left_v).map_err(|_| { + PipelineError::UnableToCast(left_v.to_string(), "Int".to_string()) + })?; + Ok(Field::Boolean(left_val < (right_v as i64))) + } Field::I128(right_v) => { let left_val = i128::from_str(left_v).map_err(|_| { PipelineError::UnableToCast(left_v.to_string(), "I128".to_string()) @@ -1062,6 +1235,7 @@ pub fn evaluate_lt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -1088,6 +1262,7 @@ pub fn evaluate_lt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -1114,6 +1289,7 @@ pub fn evaluate_lt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -1140,6 +1316,7 @@ pub fn evaluate_lt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -1201,6 +1378,12 @@ pub fn evaluate_gt( })?; Ok(Field::Boolean(left_v & !right_v_b)) } + Field::Int8(right_v) => { + let right_v_b = bool::from_str(right_v.to_string().as_str()).map_err(|_| { + PipelineError::UnableToCast(format!("{}", right_v), "Bool".to_string()) + })?; + Ok(Field::Boolean(left_v & !right_v_b)) + } // left: Bool, right: I128 Field::I128(right_v) => { let right_v_b = bool::from_str(right_v.to_string().as_str()).map_err(|_| { @@ -1233,6 +1416,8 @@ pub fn evaluate_gt( Field::Int(left_v) => match right_p { // left: Int, right: Int Field::Int(right_v) => Ok(Field::Boolean(left_v > right_v)), + Field::Int8(right_v) => Ok(Field::Boolean(left_v > (right_v as i64))), + // left: Int, right: I128 Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) > right_v)), // left: Int, right: UInt @@ -1275,9 +1460,57 @@ pub fn evaluate_gt( ">".to_string(), )), }, + Field::Int8(left_v) => match right_p { + // left: Int, right: Int + Field::Int(right_v) => Ok(Field::Boolean((left_v as i64) > right_v)), + Field::Int8(right_v) => Ok(Field::Boolean(left_v > right_v)), + + // left: Int, right: I128 + Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) > right_v)), + // left: Int, right: UInt + Field::UInt(right_v) => Ok(Field::Boolean((left_v as i64) > (right_v as i64))), + // left: Int, right: U128 + Field::U128(right_v) => Ok(Field::Boolean((left_v as i64) > (right_v as i64))), + // left: Int, right: Float + Field::Float(right_v) => Ok(Field::Boolean(left_v as f64 > *right_v)), + // left: Int, right: Decimal + Field::Decimal(right_v) => { + let left_v_d = Decimal::from_i64(left_v as i64).ok_or( + PipelineError::UnableToCast(format!("{}", left_v), "Decimal".to_string()), + )?; + Ok(Field::Boolean(left_v_d > right_v)) + } + // left: Int, right: String or Text + Field::String(right_v) | Field::Text(right_v) => { + let right_v_b = i64::from_str(right_v.as_str()).map_err(|_| { + PipelineError::UnableToCast(right_v.to_string(), "Int".to_string()) + })?; + Ok(Field::Boolean((left_v as i64) > right_v_b)) + } + // left: Int, right: Duration + Field::Duration(right_v) => { + let left_v_b = + DozerDuration(Duration::from_nanos(left_v as u64), TimeUnit::Nanoseconds); + Ok(Field::Boolean(left_v_b > right_v)) + } + // left: Int, right: Null + Field::Null => Ok(Field::Null), + Field::Boolean(_) + | Field::Binary(_) + | Field::Timestamp(_) + | Field::Date(_) + | Field::Json(_) + | Field::Point(_) => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + ">".to_string(), + )), + }, Field::I128(left_v) => match right_p { // left: I128, right: Int Field::Int(right_v) => Ok(Field::Boolean(left_v > (right_v as i128))), + Field::Int8(right_v) => Ok(Field::Boolean(left_v > (right_v as i128))), + // left: I128, right: I128 Field::I128(right_v) => Ok(Field::Boolean(left_v > right_v)), // left: I128, right: UInt @@ -1323,6 +1556,8 @@ pub fn evaluate_gt( Field::UInt(left_v) => match right_p { // left: UInt, right: Int Field::Int(right_v) => Ok(Field::Boolean((left_v as i64) > right_v)), + Field::Int8(right_v) => Ok(Field::Boolean((left_v as i64) > (right_v as i64))), + // left: UInt, right: I128 Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) > right_v)), // left: UInt, right: UInt @@ -1366,6 +1601,8 @@ pub fn evaluate_gt( Field::U128(left_v) => match right_p { // left: U128, right: Int Field::Int(right_v) => Ok(Field::Boolean((left_v as i128) > (right_v as i128))), + Field::Int8(right_v) => Ok(Field::Boolean((left_v as i128) > (right_v as i128))), + // left: U128, right: I128 Field::I128(right_v) => Ok(Field::Boolean((left_v as i128) > right_v)), // left: U128, right: UInt @@ -1416,6 +1653,7 @@ pub fn evaluate_gt( Field::U128(right_v) => Ok(Field::Boolean(*left_v > right_v as f64)), // left: Float, right: Int Field::Int(right_v) => Ok(Field::Boolean(*left_v > right_v as f64)), + Field::Int8(right_v) => Ok(Field::Boolean(*left_v > right_v as f64)), // left: Float, right: I128 Field::I128(right_v) => Ok(Field::Boolean(*left_v > right_v as f64)), // left: Float, right: Decimal @@ -1463,6 +1701,12 @@ pub fn evaluate_gt( )?; Ok(Field::Boolean(left_v > right_v_d)) } + Field::Int8(right_v) => { + let right_v_d = Decimal::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast(format!("{}", right_v), "Decimal".to_string()), + )?; + Ok(Field::Boolean(left_v > right_v_d)) + } // left: Decimal, right: I128 Field::I128(right_v) => { let right_v_d = Decimal::from_i128(right_v).ok_or( @@ -1530,6 +1774,12 @@ pub fn evaluate_gt( })?; Ok(Field::Boolean(left_val > right_v)) } + Field::Int8(right_v) => { + let left_val = i64::from_str(left_v).map_err(|_| { + PipelineError::UnableToCast(left_v.to_string(), "Int".to_string()) + })?; + Ok(Field::Boolean(left_val > (right_v as i64))) + } Field::I128(right_v) => { let left_val = i128::from_str(left_v).map_err(|_| { PipelineError::UnableToCast(left_v.to_string(), "I128".to_string()) @@ -1596,6 +1846,7 @@ pub fn evaluate_gt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -1622,6 +1873,7 @@ pub fn evaluate_gt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -1648,6 +1900,7 @@ pub fn evaluate_gt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -1674,6 +1927,7 @@ pub fn evaluate_gt( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) diff --git a/dozer-sql/expression/src/execution.rs b/dozer-sql/expression/src/execution.rs index b9a642fd39..1c2891b234 100644 --- a/dozer-sql/expression/src/execution.rs +++ b/dozer-sql/expression/src/execution.rs @@ -768,6 +768,7 @@ fn validate_avg(args: &[Expression], schema: &Schema) -> Result FieldType::Decimal, FieldType::U128 => FieldType::Decimal, FieldType::Int => FieldType::Decimal, + FieldType::Int8 => FieldType::Decimal, FieldType::I128 => FieldType::Decimal, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, @@ -821,6 +822,7 @@ fn validate_max(args: &[Expression], schema: &Schema) -> Result FieldType::UInt, FieldType::U128 => FieldType::U128, FieldType::Int => FieldType::Int, + FieldType::Int8 => FieldType::Int8, FieldType::I128 => FieldType::I128, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, @@ -866,6 +868,7 @@ fn validate_min(args: &[Expression], schema: &Schema) -> Result FieldType::UInt, FieldType::U128 => FieldType::U128, FieldType::Int => FieldType::Int, + FieldType::Int8 => FieldType::Int8, FieldType::I128 => FieldType::I128, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, @@ -912,6 +915,7 @@ fn validate_max_append_only(args: &[Expression], schema: &Schema) -> Result FieldType::UInt, FieldType::U128 => FieldType::U128, FieldType::Int => FieldType::Int, + FieldType::Int8 => FieldType::Int8, FieldType::I128 => FieldType::I128, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, @@ -957,6 +961,7 @@ fn validate_min_append_only(args: &[Expression], schema: &Schema) -> Result FieldType::UInt, FieldType::U128 => FieldType::U128, FieldType::Int => FieldType::Int, + FieldType::Int8 => FieldType::Int8, FieldType::I128 => FieldType::I128, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, @@ -1002,6 +1007,7 @@ fn validate_sum(args: &[Expression], schema: &Schema) -> Result FieldType::UInt, FieldType::U128 => FieldType::U128, FieldType::Int => FieldType::Int, + FieldType::Int8 => FieldType::Int8, FieldType::I128 => FieldType::I128, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, @@ -1045,6 +1051,7 @@ fn validate_max_value(args: &[Expression], schema: &Schema) -> Result FieldType::UInt, FieldType::U128 => FieldType::U128, FieldType::Int => FieldType::Int, + FieldType::Int8 => FieldType::Int8, FieldType::I128 => FieldType::I128, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, @@ -1091,6 +1098,7 @@ fn validate_min_value(args: &[Expression], schema: &Schema) -> Result FieldType::UInt, FieldType::U128 => FieldType::U128, FieldType::Int => FieldType::Int, + FieldType::Int8 => FieldType::Int8, FieldType::I128 => FieldType::I128, FieldType::Float => FieldType::Float, FieldType::Decimal => FieldType::Decimal, diff --git a/dozer-sql/expression/src/logical.rs b/dozer-sql/expression/src/logical.rs index b7f80fd76e..63a9017840 100644 --- a/dozer-sql/expression/src/logical.rs +++ b/dozer-sql/expression/src/logical.rs @@ -20,6 +20,7 @@ pub fn evaluate_and( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::String(_) @@ -39,6 +40,7 @@ pub fn evaluate_and( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::String(_) @@ -55,6 +57,7 @@ pub fn evaluate_and( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::String(_) @@ -85,6 +88,7 @@ pub fn evaluate_or( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::String(_) @@ -104,6 +108,7 @@ pub fn evaluate_or( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::String(_) @@ -119,6 +124,7 @@ pub fn evaluate_or( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::String(_) @@ -146,6 +152,7 @@ pub fn evaluate_not( Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::String(_) diff --git a/dozer-sql/expression/src/mathematical/mod.rs b/dozer-sql/expression/src/mathematical/mod.rs index ea08f7e94a..fa084be15e 100644 --- a/dozer-sql/expression/src/mathematical/mod.rs +++ b/dozer-sql/expression/src/mathematical/mod.rs @@ -73,6 +73,7 @@ macro_rules! define_math_operator { Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -155,6 +156,7 @@ macro_rules! define_math_operator { Field::UInt(_) | Field::U128(_) | Field::Int(_) + | Field::Int8(_) | Field::I128(_) | Field::Float(_) | Field::Boolean(_) @@ -203,6 +205,39 @@ macro_rules! define_math_operator { ))), } } + + Field::Int8(v) => { + let right_v = v as i64; + return match $op { + "/" | "%" => { + if right_v == 0_i64 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + left_v, + OrderedFloat::::from_i64(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + &_ => Ok(Field::Float($fct( + left_v, + OrderedFloat::::from_i64(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))), + }; + } + // left: Float, right: I128 Field::I128(right_v) => { return match $op { @@ -414,6 +449,45 @@ macro_rules! define_math_operator { )), }; } + + Field::Int8(v) => { + let right_v = v as i64; + return match $op { + // When Int / Int division happens + "/" => { + if right_v == 0_i64 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + OrderedFloat::::from_i64(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => { + Ok(Field::Int($fct(Wrapping(left_v), Wrapping(right_v)).0)) + } + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + // left: Int, right: I128 Field::I128(right_v) => { return match $op { @@ -623,6 +697,297 @@ macro_rules! define_math_operator { $op.to_string(), )), }, + Field::Int8(v) => { + let left_v = v as i64; + return match right_p { + // left: Int, right: Int + Field::Int(right_v) => { + return match $op { + // When Int / Int division happens + "/" => { + if right_v == 0_i64 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + OrderedFloat::::from_i64(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => { + Ok(Field::Int($fct(Wrapping(left_v), Wrapping(right_v)).0)) + } + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + + Field::Int8(v) => { + let right_v = v as i64; + return match $op { + // When Int / Int division happens + "/" => { + if right_v == 0_i64 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + OrderedFloat::::from_i64(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => { + Ok(Field::Int($fct(Wrapping(left_v), Wrapping(right_v)).0)) + } + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + + // left: Int, right: I128 + Field::I128(right_v) => { + return match $op { + // When Int / I128 division happens + "/" => { + if right_v == 0_i128 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + OrderedFloat::::from_i128(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => Ok(Field::I128( + $fct(Wrapping(left_v as i128), Wrapping(right_v)).0, + )), + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + // left: Int, right: UInt + Field::UInt(right_v) => { + return match $op { + // When Int / UInt division happens + "/" => { + if right_v == 0_u64 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + OrderedFloat::::from_u64(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => Ok(Field::Int( + $fct(Wrapping(left_v), Wrapping(right_v as i64)).0, + )), + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + // left: Int, right: U128 + Field::U128(right_v) => { + return match $op { + // When Int / U128 division happens + "/" => { + if right_v == 0_u128 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + OrderedFloat::::from_u128(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => Ok(Field::I128( + $fct(Wrapping(left_v as i128), Wrapping(right_v as i128)).0, + )), + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + // left: Int, right: Float + Field::Float(right_v) => { + return match $op { + "/" | "%" => { + if right_v == 0_f64 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + right_v, + ))) + } + } + &_ => Ok(Field::Float($fct( + OrderedFloat::::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + right_v, + ))), + } + } + + // left: Int, right: Decimal + Field::Decimal(right_v) => { + return match $op { + "/" => Ok(Field::Decimal( + Decimal::from_i64(left_v) + .ok_or(PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ))? + .checked_div(right_v) + .ok_or(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + ))?, + )), + "%" => Ok(Field::Decimal( + Decimal::from_i64(left_v) + .ok_or(PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ))? + .checked_rem(right_v) + .ok_or(PipelineError::SqlError( + OperationError::ModuloByZeroOrOverflow, + ))?, + )), + "*" => Ok(Field::Decimal( + Decimal::from_i64(left_v) + .ok_or(PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ))? + .checked_mul(right_v) + .ok_or(PipelineError::SqlError( + OperationError::MultiplicationOverflow, + ))?, + )), + "+" | "-" => Ok(Field::Decimal($fct( + Decimal::from_i64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ), + )?, + right_v, + ))), + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + } + } + // left: Int, right: Null + Field::Null => Ok(Field::Null), + Field::Boolean(_) + | Field::String(_) + | Field::Text(_) + | Field::Binary(_) + | Field::Timestamp(_) + | Field::Date(_) + | Field::Json(_) + | Field::Point(_) + | Field::Duration(_) => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } Field::I128(left_v) => match right_p { // left: I128, right: Int Field::Int(right_v) => { @@ -661,6 +1026,42 @@ macro_rules! define_math_operator { )), }; } + Field::Int8(right_v) => { + return match $op { + // When I128 / Int division happens + "/" => { + if right_v == 0_i8 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float($fct( + OrderedFloat::::from_i128(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + OrderedFloat::::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + ))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => Ok(Field::I128( + $fct(Wrapping(left_v), Wrapping(right_v as i128)).0, + )), + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } // left: I128, right: I128 Field::I128(right_v) => { return match $op { @@ -886,6 +1287,43 @@ macro_rules! define_math_operator { )), }; } + Field::Int8(right_v) => { + return match $op { + // When UInt / Int division happens + "/" => { + if right_v == 0_i8 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float(OrderedFloat($fct( + f64::from_u64(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + f64::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + )))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => Ok(Field::Int( + $fct(Wrapping(left_v as i64), Wrapping(right_v as i64)).0, + )), + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + // left: UInt, right: I128 Field::I128(right_v) => { return match $op { @@ -1156,6 +1594,44 @@ macro_rules! define_math_operator { )), }; } + + Field::Int8(right_v) => { + return match $op { + // When U128 / Int division happens + "/" => { + if right_v == 0_i8 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Float(OrderedFloat($fct( + f64::from_u128(left_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "f64".to_string(), + ), + )?, + f64::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast( + format!("{}", right_v), + "f64".to_string(), + ), + )?, + )))) + } + } + // When it's not division operation + "+" | "-" | "*" | "%" => Ok(Field::I128( + $fct(Wrapping(left_v as i128), Wrapping(right_v as i128)).0, + )), + &_ => Err(PipelineError::InvalidTypeComparison( + left_p, + right_p, + $op.to_string(), + )), + }; + } + // left: U128, right: I128 Field::I128(right_v) => { return match $op { @@ -1384,6 +1860,28 @@ macro_rules! define_math_operator { )) } } + Field::Int8(v) => { + let right_v = v as i64; + if right_v == 0_i64 { + Err(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + )) + } else { + Ok(Field::Decimal( + left_v + .checked_div(Decimal::from_i64(right_v).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ), + )?) + .ok_or(PipelineError::SqlError( + OperationError::DivisionByZeroOrOverflow, + ))?, + )) + } + } + // left: Decimal, right: I128 Field::I128(right_v) => { if right_v == 0_i128 { @@ -1506,6 +2004,18 @@ macro_rules! define_math_operator { OperationError::ModuloByZeroOrOverflow, ))?, )), + Field::Int8(right_v) => Ok(Field::Decimal( + left_v + .checked_rem(Decimal::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ), + )?) + .ok_or(PipelineError::SqlError( + OperationError::ModuloByZeroOrOverflow, + ))?, + )), // left: Decimal, right: I128 Field::I128(right_v) => Ok(Field::Decimal( left_v @@ -1596,6 +2106,18 @@ macro_rules! define_math_operator { OperationError::MultiplicationOverflow, ))?, )), + Field::Int8(right_v) => Ok(Field::Decimal( + left_v + .checked_mul(Decimal::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ), + )?) + .ok_or(PipelineError::SqlError( + OperationError::MultiplicationOverflow, + ))?, + )), // left: Decimal, right: I128 Field::I128(right_v) => Ok(Field::Decimal( left_v @@ -1683,6 +2205,15 @@ macro_rules! define_math_operator { ), )?, ))), + Field::Int8(right_v) => Ok(Field::Decimal($fct( + left_v, + Decimal::from_i64(right_v as i64).ok_or( + PipelineError::UnableToCast( + format!("{}", left_v), + "Decimal".to_string(), + ), + )?, + ))), // left: Decimal, right: I128 Field::I128(right_v) => Ok(Field::Decimal($fct( left_v, @@ -1785,6 +2316,8 @@ pub fn evaluate_plus( Field::UInt(v) => Ok(Field::UInt(v)), Field::U128(v) => Ok(Field::U128(v)), Field::Int(v) => Ok(Field::Int(v)), + Field::Int8(v) => Ok(Field::Int8(v)), + Field::I128(v) => Ok(Field::I128(v)), Field::Float(v) => Ok(Field::Float(v)), Field::Decimal(v) => Ok(Field::Decimal(v)), @@ -1814,6 +2347,7 @@ pub fn evaluate_minus( Field::UInt(v) => Ok(Field::UInt(v)), Field::U128(v) => Ok(Field::U128(v)), Field::Int(v) => Ok(Field::Int(-v)), + Field::Int8(v) => Ok(Field::Int8(-v)), Field::I128(v) => Ok(Field::I128(-v)), Field::Float(v) => Ok(Field::Float(-v)), Field::Decimal(v) => Ok(Field::Decimal(v.neg())), diff --git a/dozer-sql/expression/src/python_udf.rs b/dozer-sql/expression/src/python_udf.rs index 011dc68772..25f5394f04 100644 --- a/dozer-sql/expression/src/python_udf.rs +++ b/dozer-sql/expression/src/python_udf.rs @@ -62,6 +62,7 @@ pub fn evaluate_py_udf( FieldType::UInt => Field::UInt(res.extract::()?), FieldType::U128 => Field::U128(res.extract::()?), FieldType::Int => Field::Int(res.extract::()?), + FieldType::Int8 => Field::Int8(res.extract::()?), FieldType::I128 => Field::I128(res.extract::()?), FieldType::Float => Field::Float(OrderedFloat::from(res.extract::()?)), FieldType::Boolean => Field::Boolean(res.extract::()?), diff --git a/dozer-sql/expression/src/scalar/number.rs b/dozer-sql/expression/src/scalar/number.rs index d08a3e1c16..787d425907 100644 --- a/dozer-sql/expression/src/scalar/number.rs +++ b/dozer-sql/expression/src/scalar/number.rs @@ -16,6 +16,7 @@ pub(crate) fn evaluate_abs( Field::UInt(u) => Ok(Field::UInt(u)), Field::U128(u) => Ok(Field::U128(u)), Field::Int(i) => Ok(Field::Int(i.abs())), + Field::Int8(i) => Ok(Field::Int8(i.abs())), Field::I128(i) => Ok(Field::I128(i.abs())), Field::Float(f) => Ok(Field::Float(f.abs())), Field::Decimal(d) => Ok(Field::Decimal(d.abs())), @@ -50,6 +51,7 @@ pub(crate) fn evaluate_round( Field::UInt(u) => places = u as i32, Field::U128(u) => places = u as i32, Field::Int(i) => places = i as i32, + Field::Int8(i) => places = i as i32, Field::I128(i) => places = i as i32, Field::Float(f) => places = f.round().0 as i32, Field::Decimal(d) => { @@ -79,6 +81,7 @@ pub(crate) fn evaluate_round( Field::UInt(u) => Ok(Field::UInt(u)), Field::U128(u) => Ok(Field::U128(u)), Field::Int(i) => Ok(Field::Int(i)), + Field::Int8(i) => Ok(Field::Int8(i)), Field::I128(i) => Ok(Field::I128(i)), Field::Float(f) => Ok(Field::Float((f * order).round() / order)), Field::Decimal(d) => Ok(Field::Decimal(d.round_dp(places as u32))), diff --git a/dozer-sql/expression/src/scalar/string.rs b/dozer-sql/expression/src/scalar/string.rs index 28589eabca..7e557fe63c 100644 --- a/dozer-sql/expression/src/scalar/string.rs +++ b/dozer-sql/expression/src/scalar/string.rs @@ -36,6 +36,7 @@ pub fn evaluate_ucase( FieldType::UInt | FieldType::U128 | FieldType::Int + | FieldType::Int8 | FieldType::I128 | FieldType::Float | FieldType::Decimal @@ -95,6 +96,7 @@ pub fn evaluate_concat( FieldType::UInt | FieldType::U128 | FieldType::Int + | FieldType::Int8 | FieldType::I128 | FieldType::Float | FieldType::Decimal @@ -176,6 +178,7 @@ pub fn evaluate_trim( FieldType::UInt | FieldType::U128 | FieldType::Int + | FieldType::Int8 | FieldType::I128 | FieldType::Float | FieldType::Decimal @@ -299,6 +302,7 @@ pub(crate) fn evaluate_chr( }) } } + Field::Int8(i) => Ok(Field::String(((i as u8) as char).to_string())), Field::I128(i) => { if i >= 0 { Ok(Field::String((((i % 256) as u8) as char).to_string())) diff --git a/dozer-sql/src/aggregation/avg.rs b/dozer-sql/src/aggregation/avg.rs index 32216ab6f3..92878cd5d7 100644 --- a/dozer-sql/src/aggregation/avg.rs +++ b/dozer-sql/src/aggregation/avg.rs @@ -24,6 +24,7 @@ impl AvgAggregator { Self { current_state: SumState { int_state: 0_i64, + int8_state: 0_i8, i128_state: 0_i128, uint_state: 0_u64, u128_state: 0_u128, @@ -102,6 +103,13 @@ fn get_average( let i_sum = sum.to_int().ok_or(InvalidValue(sum.to_string())).unwrap(); Ok(Field::Int(i_sum.div_wrapping(*current_count as i64))) } + FieldType::Int8 => { + if *current_count == 0 { + return Ok(Field::Null); + } + let i_sum = sum.to_int8().ok_or(InvalidValue(sum.to_string())).unwrap(); + Ok(Field::Int8(i_sum.div_wrapping(*current_count as i8))) + } FieldType::I128 => { if *current_count == 0 { return Ok(Field::Null); diff --git a/dozer-sql/src/aggregation/count.rs b/dozer-sql/src/aggregation/count.rs index 31e3eeaa1c..d653f68ba7 100644 --- a/dozer-sql/src/aggregation/count.rs +++ b/dozer-sql/src/aggregation/count.rs @@ -49,6 +49,7 @@ fn get_count(count: u64, return_type: Option) -> Result Ok(Field::UInt(count)), FieldType::U128 => Ok(Field::U128(count as u128)), FieldType::Int => Ok(Field::Int(count as i64)), + FieldType::Int8 => Ok(Field::Int8(count as i8)), FieldType::I128 => Ok(Field::I128(count as i128)), FieldType::Float => Ok(Field::Float(OrderedFloat::from(count as f64))), FieldType::Decimal => Ok(Field::Decimal(calculate_err_type!( diff --git a/dozer-sql/src/aggregation/max_append_only.rs b/dozer-sql/src/aggregation/max_append_only.rs index 2da8719770..d586ff5eae 100644 --- a/dozer-sql/src/aggregation/max_append_only.rs +++ b/dozer-sql/src/aggregation/max_append_only.rs @@ -87,6 +87,17 @@ impl Aggregator for MaxAppendOnlyAggregator { self.update_state(Field::Int(new_val)); } } + FieldType::Int8 => { + let new_val = calculate_err_field!(val.to_int8(), MaxAppendOnly, val); + let max_val = match cur_max { + Field::Null => i8::MIN, + _ => calculate_err_field!(cur_max.to_int8(), MaxAppendOnly, val), + }; + + if new_val > max_val { + self.update_state(Field::Int8(new_val)); + } + } FieldType::I128 => { let new_val = calculate_err_field!(val.to_i128(), MaxAppendOnly, val); let max_val = match cur_max { diff --git a/dozer-sql/src/aggregation/min_append_only.rs b/dozer-sql/src/aggregation/min_append_only.rs index 39b22034f8..6dbd471701 100644 --- a/dozer-sql/src/aggregation/min_append_only.rs +++ b/dozer-sql/src/aggregation/min_append_only.rs @@ -87,6 +87,17 @@ impl Aggregator for MinAppendOnlyAggregator { self.update_state(Field::Int(new_val)); } } + FieldType::Int8 => { + let new_val = calculate_err_field!(val.to_int8(), MinAppendOnly, val); + let min_val = match cur_min { + Field::Null => i8::MAX, + _ => calculate_err_field!(cur_min.to_int8(), MinAppendOnly, val), + }; + + if new_val < min_val { + self.update_state(Field::Int8(new_val)); + } + } FieldType::I128 => { let new_val = calculate_err_field!(val.to_i128(), MinAppendOnly, val); let min_val = match cur_min { diff --git a/dozer-sql/src/aggregation/sum.rs b/dozer-sql/src/aggregation/sum.rs index 460181722b..03c16b7482 100644 --- a/dozer-sql/src/aggregation/sum.rs +++ b/dozer-sql/src/aggregation/sum.rs @@ -17,6 +17,7 @@ pub struct SumAggregator { #[derive(Debug, bincode::Encode, bincode::Decode)] pub struct SumState { pub(crate) int_state: i64, + pub(crate) int8_state: i8, pub(crate) i128_state: i128, pub(crate) uint_state: u64, pub(crate) u128_state: u128, @@ -32,6 +33,7 @@ impl SumAggregator { current_state: SumState { int_state: 0_i64, i128_state: 0_i128, + int8_state: 0_i8, uint_state: 0_u64, u128_state: 0_u128, float_state: 0_f64, @@ -112,6 +114,20 @@ pub fn get_sum( } Ok(Field::Int(current_state.int_state)) } + FieldType::Int8 => { + if decr { + for field in fields { + let val = calculate_err_field!(field.to_int8(), Sum, field); + current_state.int8_state -= val; + } + } else { + for field in fields { + let val = calculate_err_field!(field.to_int8(), Sum, field); + current_state.int8_state += val; + } + } + Ok(Field::Int(current_state.int_state)) + } FieldType::I128 => { if decr { for field in fields { diff --git a/dozer-tests/src/sql_tests/helper/mapper.rs b/dozer-tests/src/sql_tests/helper/mapper.rs index 21ce4cf6d0..0d241dc331 100644 --- a/dozer-tests/src/sql_tests/helper/mapper.rs +++ b/dozer-tests/src/sql_tests/helper/mapper.rs @@ -211,6 +211,7 @@ fn get_record_from_json(data: String, schema: &Schema) -> Record { let value = match field_definition.typ { FieldType::UInt | FieldType::Int + | FieldType::Int8 | FieldType::Float | FieldType::Boolean | FieldType::String diff --git a/dozer-types/src/arrow_types/to_arrow.rs b/dozer-types/src/arrow_types/to_arrow.rs index b19b926a02..877174ceb3 100644 --- a/dozer-types/src/arrow_types/to_arrow.rs +++ b/dozer-types/src/arrow_types/to_arrow.rs @@ -153,6 +153,7 @@ pub fn map_field_type(typ: FieldType) -> DataType { FieldType::UInt => DataType::UInt64, FieldType::U128 => DataType::Utf8, FieldType::Int => DataType::Int64, + FieldType::Int8 => DataType::Int64, FieldType::I128 => DataType::Utf8, FieldType::Float => DataType::Float64, FieldType::Boolean => DataType::Boolean, diff --git a/dozer-types/src/grpc_types.rs b/dozer-types/src/grpc_types.rs index 947fca6af4..9d402e0554 100644 --- a/dozer-types/src/grpc_types.rs +++ b/dozer-types/src/grpc_types.rs @@ -106,6 +106,9 @@ pub mod conversions { Field::Int(n) => Value { value: Some(value::Value::IntValue(n)), }, + Field::Int8(n) => Value { + value: Some(value::Value::IntValue(n as i64)), + }, Field::I128(n) => Value { value: Some(value::Value::Int128Value(n.to_string())), }, @@ -163,6 +166,7 @@ pub mod conversions { FieldType::UInt => Type::UInt, FieldType::U128 => Type::U128, FieldType::Int => Type::Int, + FieldType::Int8 => Type::Int, FieldType::I128 => Type::I128, FieldType::Float => Type::Float, FieldType::Boolean => Type::Boolean, diff --git a/dozer-types/src/helper.rs b/dozer-types/src/helper.rs index 508d6a40e0..7c96e1f655 100644 --- a/dozer-types/src/helper.rs +++ b/dozer-types/src/helper.rs @@ -38,6 +38,9 @@ pub fn json_value_to_field( FieldType::Int => serde_json::from_value(value) .map_err(DeserializationError::Json) .map(Field::Int), + FieldType::Int8 => serde_json::from_value(value) + .map_err(DeserializationError::Json) + .map(Field::Int), FieldType::I128 => match value { Value::String(str) => return Field::from_str(str.as_str(), typ, nullable), _ => Err(DeserializationError::Custom( @@ -182,6 +185,20 @@ impl Field { }) } } + FieldType::Int8 => { + if nullable && (value.is_empty() || value == "null") { + Ok(Field::Null) + } else { + value + .parse::() + .map(Field::Int) + .map_err(|_| TypeError::InvalidFieldValue { + field_type: typ, + nullable, + value: value.to_string(), + }) + } + } FieldType::I128 => { if nullable && (value.is_empty() || value == "null") { Ok(Field::Null) diff --git a/dozer-types/src/json_types.rs b/dozer-types/src/json_types.rs index e556697190..96ce0cf640 100644 --- a/dozer-types/src/json_types.rs +++ b/dozer-types/src/json_types.rs @@ -108,6 +108,7 @@ pub fn field_to_json_value(field: Field) -> JsonValue { Field::UInt(n) => n.into(), Field::U128(n) => n.to_string().into(), Field::Int(n) => n.into(), + Field::Int8(n) => n.into(), Field::I128(n) => n.to_string().into(), Field::Float(n) => n.0.into(), Field::Boolean(b) => b.into(), diff --git a/dozer-types/src/types/field.rs b/dozer-types/src/types/field.rs index ff643fc301..af8d9671ad 100644 --- a/dozer-types/src/types/field.rs +++ b/dozer-types/src/types/field.rs @@ -24,6 +24,7 @@ pub enum Field { UInt(u64), U128(u128), Int(i64), + Int8(i8), I128(i128), Float(#[cfg_attr(feature= "arbitrary", arbitrary(with = arbitrary_float))] OrderedFloat), Boolean(bool), @@ -139,6 +140,7 @@ impl bincode::Encode for Field { Field::UInt(v) => v.encode(encoder), Field::U128(v) => v.encode(encoder), Field::Int(v) => v.encode(encoder), + Field::Int8(v) => v.encode(encoder), Field::I128(v) => v.encode(encoder), Field::Float(v) => v.encode(encoder), Field::Boolean(v) => v.encode(encoder), @@ -255,6 +257,7 @@ impl Field { Field::UInt(_) => 8, Field::U128(_) => 16, Field::Int(_) => 8, + Field::Int8(_) => 8, Field::I128(_) => 16, Field::Float(_) => 8, Field::Boolean(_) => 1, @@ -277,6 +280,7 @@ impl Field { Field::UInt(i) => Cow::Owned(i.to_be_bytes().into()), Field::U128(i) => Cow::Owned(i.to_be_bytes().into()), Field::Int(i) => Cow::Owned(i.to_be_bytes().into()), + Field::Int8(i) => Cow::Owned(i.to_be_bytes().into()), Field::I128(i) => Cow::Owned(i.to_be_bytes().into()), Field::Float(f) => Cow::Owned(f.to_be_bytes().into()), Field::Boolean(b) => Cow::Owned(if *b { [1] } else { [0] }.into()), @@ -392,6 +396,7 @@ impl Field { Field::Point(_) => 13, Field::Duration(_) => 14, Field::Null => 15, + Field::Int8(_) => 16, } } @@ -400,6 +405,7 @@ impl Field { Field::UInt(_) => Some(FieldType::UInt), Field::U128(_) => Some(FieldType::U128), Field::Int(_) => Some(FieldType::Int), + Field::Int8(_) => Some(FieldType::Int), Field::I128(_) => Some(FieldType::I128), Field::Float(_) => Some(FieldType::Float), Field::Boolean(_) => Some(FieldType::Boolean), @@ -611,6 +617,26 @@ impl Field { } } + pub fn to_int8(&self) -> Option { + match self { + Field::UInt(u) => i8::from_u64(*u), + Field::U128(u) => i8::from_u128(*u), + Field::Int(i) => Some((*i).try_into().unwrap()), + Field::I128(i) => i8::from_i128(*i), + Field::Float(f) => i8::from_f64(f.0), + Field::Decimal(d) => d.to_i8(), + Field::String(s) => s.parse::().ok(), + Field::Text(s) => s.parse::().ok(), + Field::Json(j) => match j.destructure_ref() { + DestructuredRef::Number(n) => Some(n.to_f64_lossy() as i8), + DestructuredRef::String(s) => s.parse::().ok(), + _ => None, + }, + Field::Null => Some(0_i8), + _ => None, + } + } + pub fn to_int(&self) -> Option { match self { Field::UInt(u) => i64::from_u64(*u), @@ -800,6 +826,7 @@ impl Display for Field { Field::UInt(u) => write!(f, "{u}"), Field::U128(u) => write!(f, "{u}"), Field::Int(i) => write!(f, "{i}"), + Field::Int8(i) => write!(f, "{i}"), Field::I128(i) => write!(f, "{i}"), Field::Float(OrderedFloat(fl)) => write!(f, "{fl}"), Field::Decimal(d) => write!(f, "{d}"), @@ -847,6 +874,7 @@ pub enum FieldType { U128, /// Signed 64-bit integer. Int, + Int8, /// Signed 128-bit integer. I128, /// 64-bit floating point number. @@ -912,6 +940,8 @@ impl Display for FieldType { FieldType::UInt => f.write_str("64-bit unsigned int"), FieldType::U128 => f.write_str("128-bit unsigned int"), FieldType::Int => f.write_str("64-bit int"), + FieldType::Int8 => f.write_str("8-bit int"), + FieldType::I128 => f.write_str("128-bit int"), FieldType::Float => f.write_str("float"), FieldType::Boolean => f.write_str("boolean"), @@ -1177,6 +1207,7 @@ impl pyo3::ToPyObject for Field { Field::UInt(val) => val.to_object(py), Field::U128(val) => val.to_object(py), Field::Int(val) => val.to_object(py), + Field::Int8(val) => val.to_object(py), Field::I128(val) => val.to_object(py), Field::Float(val) => val.0.to_object(py), Field::Decimal(val) => val.to_f64().unwrap().to_object(py),