From SQL to SPL:Get the initial date using the total
A certain database table records the planned inbound quantity and total inventory after inbound on specific dates, such as the planned inbound quantity of 0.6 on February 26th, resulting in a total inventory of 3.
ITEM |
LOC |
NEEDDATE |
QTY |
CUSTQTY |
ABC |
XYZ |
2024-02-13 00:00:00 |
4.8 |
3 |
ABC |
XYZ |
2024-02-14 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-15 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-16 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-19 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-20 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-21 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-22 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-23 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-26 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-27 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-28 00:00:00 |
0.6 |
3 |
ABC |
XYZ |
2024-02-29 00:00:00 |
0.6 |
3 |
Now, based on the given date, we need to use the total inventory to deduce the initial date, which is the day when there is zero or negative inventory. We need to add the daily consumption of UPDATED_QTY and the original inventory UPDATED_CUSTQTY. For example, given February 26th, it can be known that the original inventory of the day before inbound was 3-0.6=2.4; The previous date was February 23rd, and the original inventory on that day was 2.4-0.6=1.8; Until February 20th, the original inventory for that day was 0.
ITEM |
LOC |
NEEDDATE |
QTY |
CUSTQTY |
UPDATED_QTY |
UPDATED_CUSTQTY |
ABC |
XYZ |
2024-02-29 00:00:00 |
0.6 |
3 |
0.6 |
|
ABC |
XYZ |
2024-02-28 00:00:00 |
0.6 |
3 |
0.6 |
|
ABC |
XYZ |
2024-02-27 00:00:00 |
0.6 |
3 |
0.6 |
|
ABC |
XYZ |
2024-02-26 00:00:00 |
0.6 |
3 |
0 |
2.4 |
ABC |
XYZ |
2024-02-23 00:00:00 |
0.6 |
3 |
0 |
1.8 |
ABC |
XYZ |
2024-02-22 00:00:00 |
0.6 |
3 |
0 |
1.2 |
ABC |
XYZ |
2024-02-21 00:00:00 |
0.6 |
3 |
0 |
0.6 |
ABC |
XYZ |
2024-02-20 00:00:00 |
0.6 |
3 |
0 |
0.0 |
ABC |
XYZ |
2024-02-19 00:00:00 |
0.6 |
3 |
0.6 |
|
ABC |
XYZ |
2024-02-16 00:00:00 |
0.6 |
3 |
0.6 |
|
ABC |
XYZ |
2024-02-15 00:00:00 |
0.6 |
3 |
0.6 |
|
ABC |
XYZ |
2024-02-14 00:00:00 |
0.6 |
3 |
0.6 |
|
ABC |
XYZ |
2024-02-13 00:00:00 |
4.8 |
3 |
4.8 |
SQL:
SELECT t.*,
LEAST(
GREATEST(
COALESCE(
SUM(
CASE WHEN needdate < TRUNC(to_date('2024-02-26')) + 1 THEN qty END
) OVER (PARTITION BY item, loc ORDER BY needdate DESC) - custqty,
qty
),
0
),
qty
) AS updated_qty,
CASE
WHEN needdate > TRUNC(to_date('2024-02-26'))
THEN NULL
WHEN SUM(
CASE WHEN needdate < TRUNC(to_date('2024-02-26')) + 1 THEN qty END
) OVER (
PARTITION BY item, loc
ORDER BY needdate DESC
ROWS BETWEEN UNBOUNDED PRECEDING AND 1 PRECEDING
) >= custqty
THEN NULL
ELSE GREATEST(
custqty
- SUM(
CASE WHEN needdate < TRUNC(to_date('2024-02-26')) + 1 THEN qty END
) OVER (PARTITION BY item, loc ORDER BY needdate DESC),
0
)
END AS updated_custqty
FROM table_name t
SQL requires multiple window functions to indirectly implement ordered calculation, and the code is complex and difficult to understand. SPL can directly represent relative or absolute position:
https://try.esproc.com/splx?3FZ
A |
|
1 |
$select *, QTY as UPDATED_QTY, null as UPDATED_CUSTQTY from table_name.txt order by NEEDDATE desc |
2 |
=A1.select(NEEDDATE<=date("2024-02-26")) |
3 |
=A2.run(UPDATED_CUSTQTY=round(if(#==1,CUSTQTY-QTY,if((t=UPDATED_CUSTQTY[-1]-QTY)>=0,t)),1), if(UPDATED_CUSTQTY>=0,UPDATED_QTY=0)) |
4 |
return A1 |
A1: Load data, sort in reverse order by date, and add two calculated columns: the consumed inventory UPDATED_QTY, with the initial value being the inbound quantity; The original inventory UPDATED_CUSTQTY has an initial value of null.
A2: Filter out records before the specified date.
A3: Modify the record: If the current record is the first one, then original inventory=total inventory - received quantity; otherwise, original inventory=previous original inventory - received quantity, with the result rounded to one decimal place. If the original inventory is greater than or equal to 0, then the received quantity after consumption is 0. [-1] represents the previous record.
Question source:https://stackoverflow.com/questions/78060568/why-i-can-not-use-lag-function-in-this-case
SPL Official Website 👉 https://www.esproc.com
SPL Feedback and Help 👉 https://www.reddit.com/r/esProcSPL
SPL Learning Material 👉 https://c.esproc.com
SPL Source Code and Package 👉 https://github.com/SPLWare/esProc
Discord 👉 https://discord.gg/2bkGwqTj
Youtube 👉 https://www.youtube.com/@esProc_SPL