urQL/docs/ref-ch07-transform.md
2023-06-01 12:02:47 -07:00

168 lines
6.4 KiB
Markdown
Executable File

# Transform
The `<transform>` statements provides a means of chaining commands on `<table-set>`s produced by any command, either by passing the resulting `<table-set>` to the next command -- similar to how CTEs work -- or by applying a set operation on two resulting `<table-set>`s. It also provides the framework for declaring `<common-table-expression>`s, which can be consumed by following commands.
```
<transform> ::=
[ WITH [ <common-table-expression> [ ,...n ] ] ]
<cmd>
[ INTO <table>
| <transform-op> [ ( ] <cmd> [ ) ]
] [ ...n ]
[ AS OF { NOW
| <timestamp>
| n { SECONDS | MINUTES | HOURS | DAYS | WEEKS | MONTHS | YEARS } AGO
| <inline-scalar>
}
]
```
```
<common-table-expression> ::= <transform> [ AS ] <alias>
```
A `<transform>` in a CTE cannot include a `WITH` clause.
```
<cmd> ::=
<delete>
| <insert>
| <merge>
| <query>
| <update>
```
A `<cmd>` is considered terminal when it operates on a `<table>`, whether it mutates `<table>` state or not. The `<query>` command by itself is never terminal.
```
<transform-op> ::= <set-op> | <pass-thru-op>
```
```
<set-op> ::=
UNION
| EXCEPT
| INTERSECT
| DIVIDED BY [ WITH REMAINDER ]
```
Set operations between two result `<table-sets>`. The left `<table-set>` represents the running result of the `<transform>` and the right `<table-set>` can be the result of nested `<cmd>`s.
**UNION**
`UNION` concatenates the left and right `<table-set>`s into one `<table-set>` of distinct rows.
**EXCEPT**
`EXCEPT` returns distinct rows from the left `<table-set>` that are not in the right `<table-set>`. Rows that are not of the same `<row-type>` are considered not matching.
**INTERSECT**
`INTERSECT` returns distinct rows that are in both the left and right `<table-set>`s. Rows that are not of the same `<row-type>` are considered not matching.
**DIVIDED BY [ WITH REMAINDER ]**
This operator performs a relational division on the left `<table-set>` as the dividend and the right `<table-set>` as divisor.
NOTE: rule for dividing union `<row-type>`s TBD.
```
<pass-thru-op> ::=
PASS-THRU
| TEE
| MULTEE
```
`<pass-thru-op>`s make the left resulting `<table-set>` available for consumption by the next `<cmd>` in the `<transform>`. The right side of the statement cannot be nested. The left `<table-set>` can be consumed by `*`, in which case column identifiers and `<row-type>` column alignments from the left `<cmd>` apply, or a list of column aliases, in which case all columns produced by the left `<cmd>` must be included in the order produced. (See the definition of `<table-set>` in the Introduction.) In other words the `<row-type>` of the left `<table-set>` applies when consumed by the right `<cmd>`.
**PASS-THRU**
The result `<table-set>` of the left sequence of `<cmd>`s in the `<transform>` is available to the next (right) `<cmd>`.
**TEE**
The result `<table-set>` of the left sequence of `<cmd>`s in the `<transform>` is available to the next (right) `<cmd>` and the left `<table-set>` is placed in order in the list of `<table-set>`s resulting from the parent `<transform>`.
**MULTEE**
The result `<table-set>` of the left sequence of `<cmd>`s in the `<transform>` is available to the next (right) `<cmd>` and the results of each `<row-type>` in the left `<table-set>` union type is placed in order in the list of `<table-set>`s resulting from the parent `<transform>`. The order of the resulting `<row-type>`s from the union type is arbitrary.
NOTE: deterministic ordering of union type results TBD.
```
<set-functions> ::=
<cmd> | <set-op>
```
API:
```
+$ transform
$:
%transform
ctes=(list <common-table-expression>)
(tree <set-functions>)
==
```
## Arguments
**`WITH [ <common-table-expression> [ ,...n ] ]`**
`<transform>`s within a CTE may not have their own `WITH` clause.
The `WITH` clause makes the result `<table-set>` of a `<transform>` statement available to the subsequent `<transform>` statements in the `WITH` clause and `<cmd>`s in the main `<transform>` by `<alias>`. `<transform>`s in the `WITH` clause cannot have their own internal `WITH`, rather any preceding CTEs are available.
When used as a `<common-table-expression>`, `<transform>` output must be a pass-thru virtual-table.
When used as a `<common-table-expression>`, `<transform>` cannot include `TEE` and `MULTEE` operators.
**`INTO <table>`**
This clause inserts the resulting `<table-set>` into `<table>`. The associated `<cmd>` is terminal. This is the only case in which `<query>` is terminal.
**`<transform-op> [ ( ] <cmd> [ ) ]`**
If `<transform-op>` is a `<set-op>` the result `<table-set>` from the left side is applied to the next (right) result or result from next nested `<cmd>`s.
If `<transform-op>` is a `<pass-thru-op>` the result `<table-set>` from the left side is available to the next `<cmd>`.
Nesting left paren `(` can only exist singly, but right paren `)` may be stacked to any depth `...)))`, so long as left and right are matching. In other words nesting can only be applied on the right side.
**`AS OF`**
The `AS OF` provides a means to "travel through time" through the state changes of the `<database>`(s). The default is the current state at execution, `NOW`.
**`<timestamp>`**
Any valid date/time before the time of execution.
**`n`**
Integer seconds, minutes, hours, days, weeks, months, or years before execution time. If months is specified and the time "lands" on a day that is beyond the last day of that month, the date defaults to the last day of the landing month.
**`<inline-scalar>`**
TBD
## Remarks
The `<transform>` command potentially results in a state change of the Obelisk agent depending on the `<cmd>` in the last step.
If a `<cmd>` is terminal it must be the last `<cmd>` in the `<transform>`, cannot be nested by parentheses, and cannot be the right side of a `<set-op>`.
## Produced Metadata
list of output `<table-set>`s in order produced (if last `<cmd>` is terminal it produces no output)
metadata from last `<cmd>`, if it was not the right side of a `<set-op>`
`@@ROWCOUNT` returns the total number of rows returned, if the last `<cmd>` is in the right side of a `<set-op>`
## Exceptions
`<table>` does not exist
mismatch of result `<row-type>` and `<table>`
`GRANT` permission on `<table>` violated
unique key violation on `<table>`
`AS OF` prior to creation of any `<table>` directly or indirectly (through a `<view>`) referenced in the `<transform>`
any exception for `<cmd>` anywhere in the `<transform>`