risingwave_frontend/expr/
session_timezone.rs1use risingwave_common::types::DataType;
16pub use risingwave_pb::expr::expr_node::Type as ExprType;
17
18pub use crate::expr::expr_rewriter::ExprRewriter;
19pub use crate::expr::function_call::FunctionCall;
20use crate::expr::{Expr, ExprImpl, ExprVisitor};
21use crate::session::current;
22
23pub struct SessionTimezone {
26 timezone: String,
27 used: bool,
29}
30
31impl ExprRewriter for SessionTimezone {
32 fn rewrite_function_call(&mut self, func_call: FunctionCall) -> ExprImpl {
33 let (func_type, inputs, ret) = func_call.decompose();
34 let inputs: Vec<ExprImpl> = inputs
35 .into_iter()
36 .map(|expr| self.rewrite_expr(expr))
37 .collect();
38 if let Some(expr) = self.with_timezone(func_type, &inputs, ret.clone()) {
39 self.mark_used();
40 expr
41 } else {
42 FunctionCall::new_unchecked(func_type, inputs, ret).into()
43 }
44 }
45}
46
47impl SessionTimezone {
48 pub fn new(timezone: String) -> Self {
49 Self {
50 timezone,
51 used: false,
52 }
53 }
54
55 pub fn timezone(&self) -> String {
56 self.timezone.clone()
57 }
58
59 pub fn used(&self) -> bool {
60 self.used
61 }
62
63 fn mark_used(&mut self) {
64 if !self.used {
65 self.used = true;
66 current::notice_to_user(format!(
67 "Your session timezone is {}. It was used in the interpretation of timestamps and dates in your query. If this is unintended, \
68 change your timezone to match that of your data's with `set timezone = [timezone]` or \
69 rewrite your query with an explicit timezone conversion, e.g. with `AT TIME ZONE`.\n",
70 self.timezone
71 ));
72 }
73 }
74
75 fn with_timezone(
77 &self,
78 func_type: ExprType,
79 inputs: &[ExprImpl],
80 return_type: DataType,
81 ) -> Option<ExprImpl> {
82 match func_type {
83 ExprType::Cast => {
98 assert_eq!(inputs.len(), 1);
99 let mut input = inputs[0].clone();
100 let input_type = input.return_type();
101 match (input_type, return_type.clone()) {
102 (DataType::Timestamptz, DataType::Varchar)
103 | (DataType::Varchar, DataType::Timestamptz) => {
104 Some(self.cast_with_timezone(input, return_type))
105 }
106 (DataType::Date, DataType::Timestamptz)
107 | (DataType::Timestamp, DataType::Timestamptz) => {
108 input = input.cast_explicit(DataType::Timestamp).unwrap();
109 Some(self.at_timezone(input))
110 }
111 (DataType::Timestamptz, DataType::Date)
112 | (DataType::Timestamptz, DataType::Time)
113 | (DataType::Timestamptz, DataType::Timestamp) => {
114 input = self.at_timezone(input);
115 input = input.cast_explicit(return_type).unwrap();
116 Some(input)
117 }
118 _ => None,
119 }
120 }
121 ExprType::Equal
130 | ExprType::NotEqual
131 | ExprType::LessThan
132 | ExprType::LessThanOrEqual
133 | ExprType::GreaterThan
134 | ExprType::GreaterThanOrEqual
135 | ExprType::IsDistinctFrom
136 | ExprType::IsNotDistinctFrom => {
137 assert_eq!(inputs.len(), 2);
138 let mut inputs = inputs.to_vec();
139 for idx in 0..2 {
140 if matches!(inputs[(idx + 1) % 2].return_type(), DataType::Timestamptz)
141 && matches!(
142 inputs[idx % 2].return_type(),
143 DataType::Date | DataType::Timestamp
144 )
145 {
146 let mut to_cast = inputs[idx % 2].clone();
147 to_cast = to_cast.cast_explicit(DataType::Timestamp).unwrap();
150 inputs[idx % 2] = self.at_timezone(to_cast);
151 return Some(
152 FunctionCall::new_unchecked(func_type, inputs, return_type).into(),
153 );
154 }
155 }
156 None
157 }
158 ExprType::Subtract | ExprType::Add => {
165 assert_eq!(inputs.len(), 2);
166 let canonical_match = matches!(inputs[0].return_type(), DataType::Timestamptz)
167 && matches!(inputs[1].return_type(), DataType::Interval);
168 let inverse_match = matches!(inputs[1].return_type(), DataType::Timestamptz)
169 && matches!(inputs[0].return_type(), DataType::Interval);
170 assert!(!(inverse_match && func_type == ExprType::Subtract)); if canonical_match || inverse_match {
172 let (orig_timestamptz, interval) =
173 if func_type == ExprType::Add && inverse_match {
174 (inputs[1].clone(), inputs[0].clone())
175 } else {
176 (inputs[0].clone(), inputs[1].clone())
177 };
178 let new_type = match func_type {
179 ExprType::Add => ExprType::AddWithTimeZone,
180 ExprType::Subtract => ExprType::SubtractWithTimeZone,
181 _ => unreachable!(),
182 };
183 let rewritten_expr = FunctionCall::new(
184 new_type,
185 vec![
186 orig_timestamptz,
187 interval,
188 ExprImpl::literal_varchar(self.timezone()),
189 ],
190 )
191 .unwrap()
192 .into();
193 return Some(rewritten_expr);
194 }
195 None
196 }
197 ExprType::DateTrunc | ExprType::Extract | ExprType::DatePart => {
200 if !(inputs.len() == 2 && inputs[1].return_type() == DataType::Timestamptz) {
201 return None;
202 }
203 assert_eq!(inputs[0].return_type(), DataType::Varchar);
204 if let ExprImpl::Literal(lit) = &inputs[0]
205 && matches!(func_type, ExprType::Extract | ExprType::DatePart)
206 && lit
207 .get_data()
208 .as_ref()
209 .is_none_or(|v| v.as_utf8().eq_ignore_ascii_case("epoch"))
210 {
211 return None;
214 }
215 let mut new_inputs = inputs.to_vec();
216 new_inputs.push(ExprImpl::literal_varchar(self.timezone()));
217 Some(FunctionCall::new(func_type, new_inputs).unwrap().into())
218 }
219 ExprType::CharToTimestamptz => {
222 if !(inputs.len() == 2
223 && inputs[0].return_type() == DataType::Varchar
224 && inputs[1].return_type() == DataType::Varchar)
225 {
226 return None;
227 }
228 let mut new_inputs = inputs.to_vec();
229 new_inputs.push(ExprImpl::literal_varchar(self.timezone()));
230 Some(FunctionCall::new(func_type, new_inputs).unwrap().into())
231 }
232 ExprType::ToChar => {
235 if !(inputs.len() == 2
236 && inputs[0].return_type() == DataType::Timestamptz
237 && inputs[1].return_type() == DataType::Varchar)
238 {
239 return None;
240 }
241 let mut new_inputs = inputs.to_vec();
242 new_inputs.push(ExprImpl::literal_varchar(self.timezone()));
243 Some(FunctionCall::new(func_type, new_inputs).unwrap().into())
244 }
245 _ => None,
246 }
247 }
248
249 fn at_timezone(&self, input: ExprImpl) -> ExprImpl {
250 FunctionCall::new(
251 ExprType::AtTimeZone,
252 vec![input, ExprImpl::literal_varchar(self.timezone.clone())],
253 )
254 .unwrap()
255 .into()
256 }
257
258 fn cast_with_timezone(&self, input: ExprImpl, return_type: DataType) -> ExprImpl {
259 FunctionCall::new_unchecked(
260 ExprType::CastWithTimeZone,
261 vec![input, ExprImpl::literal_varchar(self.timezone.clone())],
262 return_type,
263 )
264 .into()
265 }
266}
267
268#[derive(Default)]
269pub struct TimestamptzExprFinder {
270 has: bool,
271}
272
273impl TimestamptzExprFinder {
274 pub fn has(&self) -> bool {
275 self.has
276 }
277}
278
279impl ExprVisitor for TimestamptzExprFinder {
280 fn visit_function_call(&mut self, func_call: &FunctionCall) {
281 if func_call.return_type() == DataType::Timestamptz {
282 self.has = true;
283 return;
284 }
285
286 for input in &func_call.inputs {
287 if input.return_type() == DataType::Timestamptz {
288 self.has = true;
289 return;
290 }
291 }
292
293 func_call
294 .inputs()
295 .iter()
296 .for_each(|expr| self.visit_expr(expr));
297 }
298}