Skip to content

Commit

Permalink
feat: support hex literal (#1030)
Browse files Browse the repository at this point in the history
## Rationale
Currently, the underlying storage supports binary data type. However,
during the parsing and planning, the binary data is not supported.

## Detailed Changes
Support binary data type during parsing and planning stage.

## Test Plan
New unit tests and integration tests.
  • Loading branch information
ShiKaiWi authored Jun 26, 2023
1 parent 1fd7857 commit cb7c907
Show file tree
Hide file tree
Showing 7 changed files with 205 additions and 47 deletions.
38 changes: 19 additions & 19 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

12 changes: 6 additions & 6 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -76,8 +76,8 @@ cluster = { path = "cluster" }
criterion = "0.3"
common_types = { path = "common_types" }
common_util = { path = "common_util" }
datafusion = { git = "https://github.com/jiacai2050/arrow-datafusion.git", rev = "13314c37020b90246db9b80f8294370c06e61018" }
datafusion-proto = { git = "https://github.com/jiacai2050/arrow-datafusion.git", rev = "13314c37020b90246db9b80f8294370c06e61018" }
datafusion = { git = "https://github.com/ceresdb/arrow-datafusion.git", rev = "acb5d97a8a8de5296989740f97db3773fe3aa45a" }
datafusion-proto = { git = "https://github.com/ceresdb/arrow-datafusion.git", rev = "acb5d97a8a8de5296989740f97db3773fe3aa45a" }
df_operator = { path = "df_operator" }
etcd-client = "0.10.3"
env_logger = "0.6"
Expand All @@ -87,10 +87,10 @@ lazy_static = "1.4.0"
log = "0.4"
logger = { path = "components/logger" }
lru = "0.7.6"
influxql-logical-planner = { git = "https://github.com/CeresDB/influxql", rev = "efbc589", package = "iox_query_influxql" }
influxql-parser = { git = "https://github.com/CeresDB/influxql", rev = "efbc589", package = "influxdb_influxql_parser" }
influxql-query = { git = "https://github.com/CeresDB/influxql", rev = "efbc589", package = "iox_query" }
influxql-schema = { git = "https://github.com/CeresDB/influxql", rev = "efbc589", package = "schema" }
influxql-logical-planner = { git = "https://github.com/CeresDB/influxql", rev = "935e037a5ad6eb142a93f3e9eb321ee72e28cbad", package = "iox_query_influxql" }
influxql-parser = { git = "https://github.com/CeresDB/influxql", rev = "935e037a5ad6eb142a93f3e9eb321ee72e28cbad", package = "influxdb_influxql_parser" }
influxql-query = { git = "https://github.com/CeresDB/influxql", rev = "935e037a5ad6eb142a93f3e9eb321ee72e28cbad", package = "iox_query" }
influxql-schema = { git = "https://github.com/CeresDB/influxql", rev = "935e037a5ad6eb142a93f3e9eb321ee72e28cbad", package = "schema" }
interpreters = { path = "interpreters" }
itertools = "0.10.5"
meta_client = { path = "meta_client" }
Expand Down
91 changes: 70 additions & 21 deletions common_types/src/datum.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
// Copyright 2022 CeresDB Project Authors. Licensed under Apache-2.0.
// Copyright 2022-2023 CeresDB Project Authors. Licensed under Apache-2.0.

//! Datum holds different kind of data
Expand All @@ -8,78 +8,78 @@ use arrow::temporal_conversions::{EPOCH_DAYS_FROM_CE, NANOSECONDS};
use ceresdbproto::schema::DataType as DataTypePb;
use chrono::{Datelike, Local, NaiveDate, NaiveTime, TimeZone, Timelike};
use serde::ser::{Serialize, Serializer};
use snafu::{Backtrace, ResultExt, Snafu};
use snafu::{Backtrace, OptionExt, ResultExt, Snafu};
use sqlparser::ast::{DataType as SqlDataType, Value};

use crate::{bytes::Bytes, hash::hash64, string::StringBytes, time::Timestamp};
use crate::{bytes::Bytes, hash::hash64, hex, string::StringBytes, time::Timestamp};

const DATE_FORMAT: &str = "%Y-%m-%d";
const TIME_FORMAT: &str = "%H:%M:%S%.3f";

#[derive(Debug, Snafu)]
pub enum Error {
#[snafu(display(
"Unsupported SQL data type, type:{}.\nBacktrace:\n{}",
sql_type,
backtrace
))]
#[snafu(display("Unsupported SQL data type, type:{sql_type}.\nBacktrace:\n{backtrace}"))]
UnsupportedDataType {
sql_type: SqlDataType,
backtrace: Backtrace,
},

#[snafu(display("Invalid double or float, err:{}.\nBacktrace:\n{}", source, backtrace))]
#[snafu(display("Invalid double or float, err:{source}.\nBacktrace:\n{backtrace}"))]
InvalidDouble {
source: std::num::ParseFloatError,
backtrace: Backtrace,
},

#[snafu(display(
"Invalid insert value, kind:{}, value:{:?}.\nBacktrace:\n{}",
kind,
value,
backtrace
"Invalid insert value, kind:{kind}, value:{value:?}.\nBacktrace:\n{backtrace}"
))]
InvalidValueType {
kind: DatumKind,
value: Value,
backtrace: Backtrace,
},
#[snafu(display("Invalid timestamp, err:{}.\nBacktrace:\n{}", source, backtrace))]

#[snafu(display("Invalid timestamp, err:{source}.\nBacktrace:\n{backtrace}"))]
InvalidTimestamp {
source: std::num::ParseIntError,
backtrace: Backtrace,
},

#[snafu(display("Invalid date, err:{}.\nBacktrace:\n{}", source, backtrace))]
#[snafu(display("Invalid date, err:{source}.\nBacktrace:\n{backtrace}"))]
InvalidDate {
source: chrono::ParseError,
backtrace: Backtrace,
},

#[snafu(display("Invalid time, err:{}.\nBacktrace:\n{}", source, backtrace))]
#[snafu(display("Invalid time, err:{source}.\nBacktrace:\n{backtrace}"))]
InvalidTimeCause {
source: chrono::ParseError,
backtrace: Backtrace,
},

#[snafu(display("Invalid time, err:{}.\nBacktrace:\n{}", source, backtrace))]
#[snafu(display("Invalid time, err:{source}.\nBacktrace:\n{backtrace}"))]
InvalidTimeHourFormat {
source: std::num::ParseIntError,
backtrace: Backtrace,
},

#[snafu(display("Invalid time, err:{}", msg))]
InvalidTimeNoCause { msg: String },
#[snafu(display("Invalid time, err:{msg}.\nBacktrace:\n{backtrace}"))]
InvalidTimeNoCause { msg: String, backtrace: Backtrace },

#[snafu(display("Invalid integer, err:{}.\nBacktrace:\n{}", source, backtrace))]
#[snafu(display("Invalid integer, err:{source}.\nBacktrace:\n{backtrace}"))]
InvalidInt {
source: std::num::ParseIntError,
backtrace: Backtrace,
},

#[snafu(display("Invalid datum byte, byte:{}.\nBacktrace:\n{}", value, backtrace))]
#[snafu(display("Invalid datum byte, byte:{value}.\nBacktrace:\n{backtrace}"))]
InvalidDatumByte { value: u8, backtrace: Backtrace },

#[snafu(display("Invalid hex value, hex_val:{hex_val}.\nBacktrace:\n{backtrace}"))]
InvalidHexValue {
hex_val: String,
backtrace: Backtrace,
},
}

pub type Result<T> = std::result::Result<T, Error>;
Expand Down Expand Up @@ -749,6 +749,10 @@ impl Datum {
(DatumKind::Varbinary, Value::DoubleQuotedString(s)) => {
Ok(Datum::Varbinary(Bytes::from(s)))
}
(DatumKind::Varbinary, Value::HexStringLiteral(s)) => {
let bytes = hex::try_decode(&s).context(InvalidHexValue { hex_val: s })?;
Ok(Datum::Varbinary(Bytes::from(bytes)))
}
(DatumKind::String, Value::DoubleQuotedString(s)) => {
Ok(Datum::String(StringBytes::from(s)))
}
Expand Down Expand Up @@ -1478,4 +1482,49 @@ mod tests {
assert!(Datum::parse_datum_time_from_str(source).is_err());
}
}

#[test]
fn test_convert_from_sql_value() {
let cases = vec![
(
Value::Boolean(false),
DatumKind::Boolean,
true,
Some(Datum::Boolean(false)),
),
(
Value::Number("100.1".to_string(), false),
DatumKind::Float,
true,
Some(Datum::Float(100.1)),
),
(
Value::SingleQuotedString("string_literal".to_string()),
DatumKind::String,
true,
Some(Datum::String(StringBytes::from_static("string_literal"))),
),
(
Value::HexStringLiteral("c70a0b".to_string()),
DatumKind::Varbinary,
true,
Some(Datum::Varbinary(Bytes::from(vec![199, 10, 11]))),
),
(
Value::EscapedStringLiteral("string_literal".to_string()),
DatumKind::String,
false,
None,
),
];

for (input, kind, succeed, expect) in cases {
let res = Datum::try_from_sql_value(&kind, input);
if succeed {
assert_eq!(res.unwrap(), expect.unwrap());
} else {
assert!(res.is_err());
}
}
}
}
Loading

0 comments on commit cb7c907

Please sign in to comment.