CONTENTS

NAME

SQL::Abstract::Reference - Reference documentation for SQL::Abstract

TERMS

Expression (expr)

The DWIM structure that's passed to most methods by default is referred to as expression syntax. If you see a variable with expr in the name, or a comment before a code block saying # expr, this is what's being described.

Abstract Query Tree (aqt)

The explicit structure that an expression is converted into before it's rendered into SQL is referred to as an abstract query tree. If you see a variable with aqt in the name, or a comment before a code block saying # aqt, this is what's being described.

SQL and Bind Values (query)

The final result of SQL::Abstract rendering is generally an SQL statement plus bind values for passing to DBI, ala:

my ($sql, @bind) = $sqla->some_method(@args);
my @hashes = @{$dbh->do($sql, { Slice => {} }, @bind)};

If you see a comment before a code block saying # query, the SQL + bind array is what's being described.

Expander

An expander subroutine is written as:

sub {
  my ($sqla, $name, $value, $k) = @_;
  ...
  return $aqt;
}

$name is the expr node type for node expanders, the op name for op expanders, and the clause name for clause expanders.

$value is the body of the thing being expanded

If an op expander is being called as the binary operator in a "hashtriple" expression, $k will be the hash key to be used as the left hand side identifier.

This can trivially be converted to an ident type AQT node with:

my $ident = $sqla->expand_expr({ -ident => $k });

Renderer

A renderer subroutine looks like:

sub {
  my ($sqla, $type, $value) = @_;
  ...
  $sqla->join_query_parts($join, @parts);
}

and can be registered on a per-type, per-op or per-clause basis.

AQT node types

An AQT node consists of a hashref with a single key, whose name is -type where 'type' is the node type, and whose value is the data for the node.

The following is an explanation of the built-in AQT type renderers; additional renderers can be registered as part of the extension system.

literal

# expr
{ -literal => [ 'SPANG(?, ?)', 1, 27 ] }

# query
SPANG(?, ?)
[ 1, 27 ]

ident

# expr
{ -ident => 'foo' }

# query
foo
[]

# expr
{ -ident => [ 'foo', 'bar' ] }

# query
foo.bar
[]

bind

# expr
{ -bind => [ 'colname', 'value' ] }

# query
?
[ 'value' ]

row

# expr
{
  -row => [ { -bind => [ 'r', 1 ] }, { -ident => [ 'clown', 'car' ] } ]
}

# query
(?, clown.car)
[ 1 ]

func

# expr
{
  -func => [ 'foo', { -ident => [ 'bar' ] }, { -bind => [ undef, 7 ] } ]
}

# query
FOO(bar, ?)
[ 7 ]

op

Standard binop:

# expr
{ -op => [
    '=', { -ident => [ 'bomb', 'status' ] },
    { -value => 'unexploded' },
] }

# query
bomb.status = ?
[ 'unexploded' ]

Prefix unop:

# expr
{ -op => [ '-', { -ident => 'foo' } ] }

# query
- foo
[]

Not as special case parenthesised unop:

# expr
{ -op => [ 'not', { -ident => 'explosive' } ] }

# query
(NOT explosive)
[]

Postfix unop: (is_null, is_not_null, asc, desc)

# expr
{ -op => [ 'is_null', { -ident => [ 'bobby' ] } ] }

# query
bobby IS NULL
[]

AND and OR:

# expr
{ -op =>
    [ 'and', { -ident => 'x' }, { -ident => 'y' }, { -ident => 'z' } ]
}

# query
( x AND y AND z )
[]

IN (and NOT IN):

# expr
{ -op => [
    'in', { -ident => 'card' }, { -bind => [ 'card', 3 ] },
    { -bind => [ 'card', 'J' ] },
] }

# query
card IN ( ?, ? )
[ 3, 'J' ]

BETWEEN (and NOT BETWEEN):

# expr
{ -op => [
    'between', { -ident => 'pints' }, { -bind => [ 'pints', 2 ] },
    { -bind => [ 'pints', 4 ] },
] }

# query
( pints BETWEEN ? AND ? )
[ 2, 4 ]

Comma (use -row for parens):

# expr
{ -op => [ ',', { -literal => [ 1 ] }, { -literal => [ 2 ] } ] }

# query
1, 2
[]

values

# expr
{ -values =>
    { -row => [ { -bind => [ undef, 1 ] }, { -bind => [ undef, 2 ] } ] }
}

# query
VALUES (?, ?)
[ 1, 2 ]

# expr
{ -values => [
    { -row => [ { -literal => [ 1 ] }, { -literal => [ 2 ] } ] },
    { -row => [ { -literal => [ 3 ] }, { -literal => [ 4 ] } ] },
] }

# query
VALUES (1, 2), (3, 4)
[]

keyword

# expr
{ -keyword => 'insert_into' }

# query
INSERT INTO
[]

statement types

AQT node types are also provided for select, insert, update and delete. These types are handled by the clauses system as discussed later.

Expressions

node expr

The simplest expression is just an AQT node:

# expr
{ -ident => [ 'foo', 'bar' ] }

# aqt
{ -ident => [ 'foo', 'bar' ] }

# query
foo.bar
[]

However, even in the case of an AQT node, the node value will be expanded if an expander has been registered for that node type:

# expr
{ -ident => 'foo.bar' }

# aqt
{ -ident => [ 'foo', 'bar' ] }

# query
foo.bar
[]

identifier hashpair types

hashtriple

# expr
{ id => { op => 'value' } }

# aqt
{ -op =>
    [ 'op', { -ident => [ 'id' ] }, { -bind => [ 'id', 'value' ] } ]
}

# query
id OP ?
[ 'value' ]

If the value is undef, attempts to convert equality and like ops to IS NULL, and inequality and not like to IS NOT NULL:

# expr
{ id => { '!=' => undef } }

# aqt
{ -op => [ 'is_not_null', { -ident => [ 'id' ] } ] }

# query
id IS NOT NULL
[]

identifier hashpair w/simple value

Equivalent to a hashtriple with an op of '='.

# expr
{ id => 'value' }

# aqt
{
  -op => [ '=', { -ident => [ 'id' ] }, { -bind => [ 'id', 'value' ] } ]
}

# query
id = ?
[ 'value' ]

(an object value will also follow this code path)

identifier hashpair w/undef RHS

Converted to IS NULL :

# expr
{ id => undef }

# aqt
{ -op => [ 'is_null', { -ident => [ 'id' ] } ] }

# query
id IS NULL
[]

(equivalent to the -is operator) :

# expr
{ id => { -is => undef } }

# aqt
{ -op => [ 'is_null', { -ident => [ 'id' ] } ] }

# query
id IS NULL
[]

identifier hashpair w/literal RHS

Directly appended to the key, remember you need to provide an operator:

# expr
{ id => \"= dont_try_this_at_home" }

# aqt
{ -literal => [ 'id = dont_try_this_at_home' ] }

# query
id = dont_try_this_at_home
[]

# expr
{ id => \[
      "= seriously(?, ?, ?, ?)",
      "use",
      "-ident",
      "and",
      "-func",
    ]
}

# aqt
{ -literal =>
    [ 'id = seriously(?, ?, ?, ?)', 'use', -ident => 'and', '-func' ]
}

# query
id = seriously(?, ?, ?, ?)
[ 'use', -ident => 'and', '-func' ]

(you may absolutely use this when there's no built-in expression type for what you need and registering a custom one would be more hassle than it's worth, but, y'know, do try and avoid it)

identifier hashpair w/arrayref value

Becomes equivalent to a -or over an arrayref of hashrefs with the identifier as key and the member of the original arrayref as the value:

# expr
{ id => [ 3, 4, { '>' => 12 } ] }

# aqt
{ -op => [
    'or',
    { -op => [ '=', { -ident => [ 'id' ] }, { -bind => [ 'id', 3 ] } ] },
    { -op => [ '=', { -ident => [ 'id' ] }, { -bind => [ 'id', 4 ] } ] },
    {
      -op => [ '>', { -ident => [ 'id' ] }, { -bind => [ 'id', 12 ] } ]
    },
] }

# query
( id = ? OR id = ? OR id > ? )
[ 3, 4, 12 ]

# expr
{ -or => [ { id => 3 }, { id => 4 }, { id => { '>' => 12 } } ] }

# aqt
{ -op => [
    'or',
    { -op => [ '=', { -ident => [ 'id' ] }, { -bind => [ 'id', 3 ] } ] },
    { -op => [ '=', { -ident => [ 'id' ] }, { -bind => [ 'id', 4 ] } ] },
    {
      -op => [ '>', { -ident => [ 'id' ] }, { -bind => [ 'id', 12 ] } ]
    },
] }

# query
( id = ? OR id = ? OR id > ? )
[ 3, 4, 12 ]

Special Case: If the first element of the arrayref is -or or -and, that's used as the top level logic op:

# expr
{ id => [ -and => { '>' => 3 }, { '<' => 6 } ] }

# aqt
{ -op => [
    'and',
    { -op => [ '>', { -ident => [ 'id' ] }, { -bind => [ 'id', 3 ] } ] },
    { -op => [ '<', { -ident => [ 'id' ] }, { -bind => [ 'id', 6 ] } ] },
] }

# query
( id > ? AND id < ? )
[ 3, 6 ]

identifier hashpair w/hashref value

Becomes equivalent to a -and over an arrayref of hashtriples constructed with the identifier as the key and each key/value pair of the original hashref as the value:

# expr
{ id => { '<' => 4, '>' => 3 } }

# aqt
{ -op => [
    'and',
    { -op => [ '<', { -ident => [ 'id' ] }, { -bind => [ 'id', 4 ] } ] },
    { -op => [ '>', { -ident => [ 'id' ] }, { -bind => [ 'id', 3 ] } ] },
] }

# query
( id < ? AND id > ? )
[ 4, 3 ]

is sugar for:

# expr
{ -and => [ { id => { '<' => 4 } }, { id => { '>' => 3 } } ] }

# aqt
{ -op => [
    'and',
    { -op => [ '<', { -ident => [ 'id' ] }, { -bind => [ 'id', 4 ] } ] },
    { -op => [ '>', { -ident => [ 'id' ] }, { -bind => [ 'id', 3 ] } ] },
] }

# query
( id < ? AND id > ? )
[ 4, 3 ]

operator hashpair types

A hashpair whose key begins with a -, or whose key consists entirely of nonword characters (thereby covering '=', '>', pg json ops, etc.) is processed as an operator hashpair.

operator hashpair w/node type

If a node type expander is registered for the key, the hashpair is treated as a "node expr".

operator hashpair w/registered op

If an expander is registered for the op name, that's run and the result returned:

# expr
{ -in => [ 'foo', 1, 2, 3 ] }

# aqt
{ -op => [
    'in', { -ident => [ 'foo' ] }, { -bind => [ undef, 1 ] },
    { -bind => [ undef, 2 ] }, { -bind => [ undef, 3 ] },
] }

# query
foo IN ( ?, ?, ? )
[ 1, 2, 3 ]

operator hashpair w/not prefix

If the op name starts -not_ this is stripped and turned into a -not wrapper around the result:

# expr
{ -not_ident => 'foo' }

# aqt
{ -op => [ 'not', { -ident => [ 'foo' ] } ] }

# query
(NOT foo)
[]

is equivalent to:

# expr
{ -not => { -ident => 'foo' } }

# aqt
{ -op => [ 'not', { -ident => [ 'foo' ] } ] }

# query
(NOT foo)
[]

operator hashpair with unknown op

If the unknown_unop_always_func option is set (which is recommended but defaults to off for backwards compatibility reasons), an unknown op expands into a -func node:

# expr
{ -count => { -ident => '*' } }

# aqt
{ -func => [ 'count', { -ident => [ '*' ] } ] }

# query
COUNT(*)
[]

If not, an unknown op will expand into a -op node.

hashref expr

A hashref with more than one pair becomes a -and over its hashpairs, i.e.

# expr
{ x => 1, y => 2 }

# aqt
{ -op => [
    'and',
    { -op => [ '=', { -ident => [ 'x' ] }, { -bind => [ 'x', 1 ] } ] },
    { -op => [ '=', { -ident => [ 'y' ] }, { -bind => [ 'y', 2 ] } ] },
] }

# query
( x = ? AND y = ? )
[ 1, 2 ]

is short hand for:

# expr
{ -and => [ { x => 1 }, { y => 2 } ] }

# aqt
{ -op => [
    'and',
    { -op => [ '=', { -ident => [ 'x' ] }, { -bind => [ 'x', 1 ] } ] },
    { -op => [ '=', { -ident => [ 'y' ] }, { -bind => [ 'y', 2 ] } ] },
] }

# query
( x = ? AND y = ? )
[ 1, 2 ]

arrayref expr

An arrayref becomes a -or over its contents. Arrayrefs, hashrefs and literals are all expanded and added to the clauses of the -or. If the arrayref contains a scalar it's treated as the key of a hashpair and the next element as the value.

# expr
[ { x => 1 }, [ { y => 2 }, { z => 3 } ], 'key', 'value', \"lit()" ]

# aqt
{ -op => [
    'or',
    { -op => [ '=', { -ident => [ 'x' ] }, { -bind => [ 'x', 1 ] } ] },
    { -op => [
        'or', {
          -op => [ '=', { -ident => [ 'y' ] }, { -bind => [ 'y', 2 ] } ]
        }, {
          -op => [ '=', { -ident => [ 'z' ] }, { -bind => [ 'z', 3 ] } ]
        },
    ] }, { -op =>
        [
          '=', { -ident => [ 'key' ] },
          { -bind => [ 'key', 'value' ] },
        ]
    },
    { -literal => [ 'lit()' ] },
] }

# query
( x = ? OR ( y = ? OR z = ? ) OR key = ? OR lit() )
[ 1, 2, 3, 'value' ]

Default Expanders

bool

Turns the old -bool syntax into the value expression, i.e.

# expr
{ -bool => { -ident => 'foo' } }

# aqt
{ -ident => [ 'foo' ] }

# query
foo
[]

behaves the same way as the now-directly-supported

# expr
{ -ident => 'foo' }

# aqt
{ -ident => [ 'foo' ] }

# query
foo
[]

row

Expands the elements of the value arrayref:

# expr
{ -row => [ 1, { -ident => 'foo' }, 2, 3 ] }

# aqt
{ -row => [
    { -bind => [ undef, 1 ] }, { -ident => [ 'foo' ] },
    { -bind => [ undef, 2 ] }, { -bind => [ undef, 3 ] },
] }

# query
(?, foo, ?, ?)
[ 1, 2, 3 ]

op

If an expander is registered for the op name, delegates to the expander; if not, expands the argument values:

# expr
{ -op => [ 'ident', 'foo.bar' ] }

# aqt
{ -ident => [ 'foo', 'bar' ] }

# query
foo.bar
[]

# expr
{ -op => [ '=', { -ident => 'foo' }, 3 ] }

# aqt
{ -op => [ '=', { -ident => [ 'foo' ] }, { -bind => [ undef, 3 ] } ] }

# query
foo = ?
[ 3 ]

func

Expands the argument values:

# expr
{ -func => [ 'coalesce', { -ident => 'thing' }, 'fallback' ] }

# aqt
{ -func => [
    'coalesce', { -ident => [ 'thing' ] },
    { -bind => [ undef, 'fallback' ] },
] }

# query
COALESCE(thing, ?)
[ 'fallback' ]

values

A hashref value is expanded as an expression:

# expr
{ -values => { -row => [ 1, 2 ] } }

# aqt
{ -values => [
    { -row => [ { -bind => [ undef, 1 ] }, { -bind => [ undef, 2 ] } ] }
] }

# query
VALUES (?, ?)
[ 1, 2 ]

An arrayref value's elements are either expressions or arrayrefs to be treated as rows:

# expr
{ -values => [ { -row => [ 1, 2 ] }, [ 3, 4 ] ] }

# aqt
{ -values => [
    { -row => [ { -bind => [ undef, 1 ] }, { -bind => [ undef, 2 ] } ] },
    { -row => [ { -bind => [ undef, 3 ] }, { -bind => [ undef, 4 ] } ] },
] }

# query
VALUES (?, ?), (?, ?)
[ 1, 2, 3, 4 ]

list

Expects a value or an arrayref of values, expands them, and returns just the expanded aqt for a single entry or a comma operator for multiple:

# expr
{ -list => [ { -ident => 'foo' } ] }

# aqt
{ -op => [ ',', { -ident => [ 'foo' ] } ] }

# query
foo
[]

# expr
{ -list => [ { -ident => 'foo' }, { -ident => 'bar' } ] }

# aqt
{ -op => [ ',', { -ident => [ 'foo' ] }, { -ident => [ 'bar' ] } ] }

# query
foo, bar
[]

between op

The RHS of between must either be a pair of exprs/plain values, or a single literal expr:

# expr
{ -between => [ 'size', 3, { -ident => 'max_size' } ] }

# aqt
{ -op => [
    'between', { -ident => [ 'size' ] }, { -bind => [ undef, 3 ] },
    { -ident => [ 'max_size' ] },
] }

# query
( size BETWEEN ? AND max_size )
[ 3 ]

# expr
{ size => { -between => [ 3, { -ident => 'max_size' } ] } }

# aqt
{ -op => [
    'between', { -ident => [ 'size' ] }, { -bind => [ 'size', 3 ] },
    { -ident => [ 'max_size' ] },
] }

# query
( size BETWEEN ? AND max_size )
[ 3 ]

# expr
{ size => { -between => \"3 AND 7" } }

# aqt
{ -op =>
    [
      'between', { -ident => [ 'size' ] },
      { -literal => [ '3 AND 7' ] },
    ]
}

# query
( size BETWEEN 3 AND 7 )
[]

not_between is also expanded:

# expr
{ size => { -not_between => [ 3, 7 ] } }

# aqt
{ -op => [
    'not_between', { -ident => [ 'size' ] },
    { -bind => [ 'size', 3 ] }, { -bind => [ 'size', 7 ] },
] }

# query
( size NOT BETWEEN ? AND ? )
[ 3, 7 ]

in op

The RHS of in/not_in is either an expr/value or an arrayref of exprs/values:

# expr
{ foo => { -in => [ 1, 2 ] } }

# aqt
{ -op => [
    'in', { -ident => [ 'foo' ] }, { -bind => [ 'foo', 1 ] },
    { -bind => [ 'foo', 2 ] },
] }

# query
foo IN ( ?, ? )
[ 1, 2 ]

# expr
{ bar => { -not_in => \"(1, 2)" } }

# aqt
{ -op =>
    [ 'not_in', { -ident => [ 'bar' ] }, { -literal => [ '1, 2' ] } ]
}

# query
bar NOT IN ( 1, 2 )
[]

A non-trivial LHS is expanded with ident as the default rather than value:

# expr
{ -in => [
    { -row => [ 'x', 'y' ] }, { -row => [ 1, 2 ] },
    { -row => [ 3, 4 ] },
] }

# aqt
{ -op => [
    'in', { -row => [ { -ident => [ 'x' ] }, { -ident => [ 'y' ] } ] },
    { -row => [ { -bind => [ undef, 1 ] }, { -bind => [ undef, 2 ] } ] },
    { -row => [ { -bind => [ undef, 3 ] }, { -bind => [ undef, 4 ] } ] },
] }

# query
(x, y) IN ( (?, ?), (?, ?) )
[ 1, 2, 3, 4 ]

and/or ops

expands the same way as a plain arrayref/hashref expression but with the logic type set to the op name.

is op

Expands is and is_not to null checks, RHS value must be undef:

# expr
{ -is => [ 'foo', undef ] }

# aqt
{ -op => [ 'is_null', { -ident => [ 'foo' ] } ] }

# query
foo IS NULL
[]

# expr
{ bar => { -is_not => undef } }

# aqt
{ -op => [ 'is_not_null', { -ident => [ 'bar' ] } ] }

# query
bar IS NOT NULL
[]

ident op

Expands a string ident to an arrayref by splitting on the configured separator, almost always '.':

# expr
{ -ident => 'foo.bar' }

# aqt
{ -ident => [ 'foo', 'bar' ] }

# query
foo.bar
[]

value op

Expands to a bind node with the currently applicable column name if known:

# expr
{ foo => { '=' => { -value => 3 } } }

# aqt
{ -op => [ '=', { -ident => [ 'foo' ] }, { -bind => [ 'foo', 3 ] } ] }

# query
foo = ?
[ 3 ]

Query Types

select

A select node accepts select, from, where and order_by clauses.

The select clause is expanded as a list expression with a -ident default:

# expr
{ -select => { _ => [ 'foo', 'bar', { -count => 'baz' } ] } }

# aqt
{ -select => { select => { -op => [
        ',', { -ident => [ 'foo' ] }, { -ident => [ 'bar' ] },
        { -func => [ 'count', { -ident => [ 'baz' ] } ] },
] } } }

# query
SELECT foo, bar, COUNT(baz)
[]

The from clause is expanded as a list expression with a -ident default:

# expr
{ -select => {
    from => [ 'schema1.table1', { -ident => [ 'schema2', 'table2' ] } ]
} }

# aqt
{ -select => { from => { -from_list => [
        { -ident => [ 'schema1', 'table1' ] },
        { -ident => [ 'schema2', 'table2' ] },
] } } }

# query
FROM schema1.table1, schema2.table2
[]

The where clause is expanded as a plain expression:

# expr
{ -select => { where => { foo => 3 } } }

# aqt
{ -select => { where => {
      -op => [ '=', { -ident => [ 'foo' ] }, { -bind => [ 'foo', 3 ] } ]
} } }

# query
WHERE foo = ?
[ 3 ]

The order_by clause expands as a list expression at top level, but a hashref element may be either an expr or a hashpair with key -asc or -desc to indicate an order by direction:

# expr
{ -select =>
    { order_by => [ 'foo', { -desc => 'bar' }, { -max => 'baz' } ] }
}

# aqt
{ -select => { order_by => { -op => [
        ',', { -ident => [ 'foo' ] }, {
          -op => [ ',', { -op => [ 'desc', { -ident => [ 'bar' ] } ] } ]
        }, { -func => [ 'max', { -ident => [ 'baz' ] } ] },
] } } }

# query
ORDER BY foo, bar DESC, MAX(baz)
[]

An insert node accepts an into/target clause, a fields clause, a values/from clause, and a returning clause.

The target clause is expanded with an ident default.

The fields clause is expanded as a list expression if an arrayref, and otherwise passed through.

The from clause may either be an expr, a literal, an arrayref of column values, or a hashref mapping colum names to values.

The returning clause is expanded as a list expr with an ident default.

# expr
{ -insert => {
    into => 'foo',
    returning => 'id',
    values => { bar => 'yay', baz => 'argh' },
} }

# aqt
{ -insert => {
    fields =>
      { -row => [ { -ident => [ 'bar' ] }, { -ident => [ 'baz' ] } ] },
    from => { -values => [ { -row => [
            { -bind => [ 'bar', 'yay' ] },
            { -bind => [ 'baz', 'argh' ] },
    ] } ] },
    returning => { -op => [ ',', { -ident => [ 'id' ] } ] },
    target => { -ident => [ 'foo' ] },
} }

# query
INSERT INTO foo (bar, baz) VALUES (?, ?) RETURNING id
[ 'yay', 'argh' ]

# expr
{ -insert => {
    fields => [ 'bar', 'baz' ],
    from => { -select => { _ => [ 'bar', 'baz' ], from => 'other' } },
    into => 'foo',
} }

# aqt
{ -insert => {
    fields => { -row => [ { -op =>
            [ ',', { -ident => [ 'bar' ] }, { -ident => [ 'baz' ] } ]
    } ] },
    from => { -select => {
        from => { -ident => [ 'other' ] },
        select => { -op =>
            [ ',', { -ident => [ 'bar' ] }, { -ident => [ 'baz' ] } ]
        },
    } },
    target => { -ident => [ 'foo' ] },
} }

# query
INSERT INTO foo (bar, baz) SELECT bar, baz FROM other
[]

update

An update node accepts update/target (either may be used at expansion time), set, where, and returning clauses.

The target clause is expanded with an ident default.

The set clause (if not already a list expr) is expanded as a hashref where the keys are identifiers to be set and the values are exprs/values.

The where clauses is expanded as a normal expr.

The returning clause is expanded as a list expr with an ident default.

# expr
{ -update => {
    _ => 'foo',
    returning => [ 'id', 'baz' ],
    set => { bar => 3, baz => { baz => { '+' => 1 } } },
    where => { -not => { -ident => 'quux' } },
} }

# aqt
{ -update => {
    returning =>
      {
        -op => [ ',', { -ident => [ 'id' ] }, { -ident => [ 'baz' ] } ]
      },
    set => { -op => [
        ',', { -op =>
            [ '=', { -ident => [ 'bar' ] }, { -bind => [ 'bar', 3 ] } ]
        }, { -op => [
            '=', { -ident => [ 'baz' ] }, { -op => [
                '+', { -ident => [ 'baz' ] },
                { -bind => [ 'baz', 1 ] },
            ] },
        ] },
    ] },
    target => { -ident => [ 'foo' ] },
    where => { -op => [ 'not', { -ident => [ 'quux' ] } ] },
} }

# query
UPDATE foo SET bar = ?, baz = baz + ? WHERE (NOT quux) RETURNING id, baz
[ 3, 1 ]

delete

delete accepts from/target, where, and returning clauses.

The target clause is expanded with an ident default.

The where clauses is expanded as a normal expr.

The returning clause is expanded as a list expr with an ident default.

# expr
{ -delete => {
    from => 'foo',
    returning => 'id',
    where => { bar => { '<' => 10 } },
} }

# aqt
{ -delete => {
    returning => { -op => [ ',', { -ident => [ 'id' ] } ] },
    target => { -op => [ ',', { -ident => [ 'foo' ] } ] },
    where => { -op =>
        [ '<', { -ident => [ 'bar' ] }, { -bind => [ 'bar', 10 ] } ]
    },
} }

# query
DELETE FROM foo WHERE bar < ? RETURNING id
[ 10 ]