I have the following table ticks
datetime | lowest_tick | tick_lower |
---|---|---|
2022-10-01 00:02:00 | 204406 | 204306 |
2022-10-01 00:03:00 | 204525 | 204425 |
2022-10-01 00:04:00 | 204414 | 204314 |
2022-10-01 00:05:00 | 204200 | 204100 |
2022-10-01 00:06:00 | 204220 | 204120 |
2022-10-01 00:07:00 | 204120 | 204020 |
What I want to get is to show the first value tick_lower_position
for tick_lower
when tick_lower <= lowest_tick
So the resulting table should look like this
datetime | lowest_tick | tick_lower | tick_lower_position |
---|---|---|---|
2022-10-01 00:02:00 | 204406 | 204306 | 204306 |
2022-10-01 00:03:00 | 204525 | 204425 | 204306 |
2022-10-01 00:04:00 | 204487 | 204387 | 204306 |
2022-10-01 00:05:00 | 204200 | 204100 | 204100 |
2022-10-01 00:06:00 | 204220 | 204120 | 204100 |
2022-10-01 00:07:00 | 204120 | 204020 | 204100 |
So far, I have tried to apply the solution provided by @the-impaler for other data.
select y.*, first_value(tick_lower)
over(partition by g order by datetime) as tick_lower_position
from (
select x.*, sum(i) over(order by datetime) as g
from (
select t.*, case when lowest_tick <
lag(tick_lower) over(order by datetime)
then 1 else 0 end as i
from t
) x
) y
But this solution doesn’t work for the current example.
As you may see in this example on db<>fidle I got the wrong value in the 3rd row. tick_lower_position
in the 3rd row still should be equal to 204306
3
Answers
Simple answer, but you might be able to just use
MIN()
as a window function along with aCASE
expression:The problem (from what I gather from the vague description and comments) does not lend itself naturally to pure SQL. Seems like a case for a procedural solution:
Create this function once:
Then call:
fiddle
I let
tick_lower_position
default toNULL
if your cited conditiontick_lower <= lowest_tick
is not met.The query you tried is almost there you just need another layer getting
least()
prior tick_lower_position and current tick_lower. See demo: