Select

select–retrieve or compute a set of values.

  1. [ with with-item [, ...] ]
  2. select expr
  3. [ filter filter-expr ]
  4. [ order by order-expr [direction] [then ...] ]
  5. [ offset offset-expr ]
  6. [ limit limit-expr ] ;

filter filter-expr

The optional filter clause, where filter-expr is any expression that has a result of type bool. The condition is evaluated for every element in the set produced by the select clause. The result of the evaluation of the filter clause is a set of boolean values. If at least one value in this set is true, the input element is included, otherwise it is eliminated from the output.

order by order-expr [direction] [then …]

The optional order by clause has this general form:

  1. order by
  2. order-expr [ asc | desc ] [ empty { first | last } ]
  3. [ then ... ]

The order by clause produces a result set sorted according to the specified expression or expressions, which are evaluated for every element of the input set.

If two elements are equal according to the leftmost expression, they are compared according to the next expression and so on. If two elements are equal according to all expressions, the resulting order is undefined.

Each expression can be an arbitrary expression that results in a value of an orderable type. Primitive types are orderable, object types are not. Additionally, the result of each expression must be an empty set or a singleton. Using an expression that may produce more elements is a compile-time error.

An optional asc or desc keyword can be added after any expression. If not specified asc is assumed by default.

If empty last is specified, then input values that produce an empty set when evaluating an expression are sorted after all other values; if empty first is specified, then they are sorted before all other values. If neither is specified, empty first is assumed when asc is specified or implied, and empty last when desc is specified.

offset offset-expr

The optional offset clause, where offset-expr is a singleton expression of an integer type. This expression is evaluated once and its result is used to skip the first element-count elements of the input set while producing the output. If element-count evaluates to an empty set, it is equivalent to offset 0, which is equivalent to omitting the offset clause. If element-count evaluates to a value that is larger then the cardinality of the input set, an empty set is produced as the result.

limit limit-expr

The optional limit clause, where limit-expr is a singleton expression of an integer type. This expression is evaluated once and its result is used to include only the first element-count elements of the input set while producing the output. If element-count evaluates to an empty set, it is equivalent to specifying no limit clause.

Description

select retrieves or computes a set of values. The data flow of a select block can be conceptualized like this:

  1. with module example
  2. select clause
  3. select
  4. expr compute a set of things
  5. optional clause
  6. filter
  7. expr filter the computed set
  8. optional clause
  9. order by
  10. expr define ordering of the filtered set
  11. optional clause
  12. offset
  13. expr slice the filtered/ordered set
  14. optional clause
  15. limit
  16. expr slice the filtered/ordered set

Please note that the order by clause defines ordering that can only be relied upon if the resulting set is not used in any other operation. select, offset and limit clauses are the only exception to that rule as they preserve the inherent ordering of the underlying set.

The first clause is select. It indicates that filter, order by, offset, or limit clauses may follow an expression, i.e. it makes an expression into a select statement. Without any of the optional clauses a (select Expr) is completely equivalent to Expr for any expression Expr.

Consider an example using the filter optional clause:

  1. with module example
  2. select User {
  3. name,
  4. owned := (select
  5. User.<owner[is Issue] {
  6. number,
  7. body
  8. }
  9. )
  10. }
  11. filter User.name like 'Alice%';

The above example retrieves a single user with a specific name. The fact that there is only one such user is a detail that can be well- known and important to the creator of the database, but otherwise non- obvious. However, forcing the cardinality to be at most 1 by using the limit clause ensures that a set with a single object or {} is returned. This way any further code that relies on the result of this query can safely assume there’s only one result available.

  1. with module example
  2. select User {
  3. name,
  4. owned := (select
  5. User.<owner[is Issue] {
  6. number,
  7. body
  8. }
  9. )
  10. }
  11. filter User.name like 'Alice%'
  12. limit 1;

Next example makes use of order by and limit clauses:

  1. with module example
  2. select Issue {
  3. number,
  4. body,
  5. due_date
  6. }
  7. filter
  8. exists Issue.due_date
  9. and
  10. Issue.status.name = 'Open'
  11. order by
  12. Issue.due_date
  13. limit 3;

The above query retrieves the top 3 open Issues with the closest due date.

Filter

The filter clause cannot affect anything aggregate-like in the preceding select clause. This is due to how filter clause works. It can be conceptualized as a function like filter($input, set of $cond), where the $input represents the value of the preceding clause, while the $cond represents the filtering condition expression. Consider the following:

  1. with module example
  2. select count(User)
  3. filter User.name like 'Alice%';

The above can be conceptualized as:

  1. with module example
  2. select _filter(
  3. count(User),
  4. User.name like 'Alice%'
  5. );

In this form it is more apparent that User is a set of argument (of count()), while User.name like 'Alice%' is also a set of argument (of filter). So the symbol User in these two expressions exists in 2 parallel scopes. Contrast it with:

  1. # This will actually only count users whose name starts with
  2. # 'Alice'.
  3. with module example
  4. select count(
  5. (select User
  6. filter User.name like 'Alice%')
  7. );
  8. # which can be represented as:
  9. with module example
  10. select count(
  11. _filter(User,
  12. User.name like 'Alice%')
  13. );

Clause signatures

Here is a summary of clauses that can be used with select:

  • A filter set of B

  • A order by set of B

  • set of A offset set of B

  • set of A limit set of B

See also

EdgeQL > Select

Cheatsheets > Selecting data