Skip to main content
Version: 4.x

Computed Columns

"Computed columns" add what appears to be an extra column (field) to the GraphQL table type, but, unlike an actual column, the value for this field is the result of calling a function defined in the PostgreSQL schema. This function will automatically be exposed to the resultant GraphQL schema as a field on the type; it can accept arguments that influence its result, and may return either a scalar, record, enum, list or a set. Sets (denoted by RETURNS SETOF ...) are exposed as connections.

Performance note: we inline these function calls into the original SELECT statement, so there's no N+1 issues - it's very efficient.

To create a function that PostGraphile will recognise as a computed column, it must obey the following rules:

  • adhere to common PostGraphile function restrictions
  • name must begin with the name of the table it applies to, followed by an underscore (_)
  • first parameter must be the table type
  • must NOT return VOID
  • must be marked as STABLE (or IMMUTABLE, though that tends to be less common)
  • must be defined in the same PostgreSQL schema as the table

For example, assuming a table called person exists, the function:

CREATE FUNCTION person_full_name(person person) RETURNS text AS $$
SELECT person.given_name || ' ' || person.family_name
$$ LANGUAGE sql STABLE;

Will create a computed column for your table named person, which can be queried like this:

{
personById(id:) {
# nodeId, id, ...

fullName # A computed column, but the client doesn’t even know!
}
}

Example

This example creates two computed columns, one returning a simple varchar and the other a connection. Note that || in PostgreSQL is string concatenation.

create table my_schema.users (
id serial not null primary key,
first_name varchar not null,
last_name varchar not null
);

create table my_schema.friendships (
user_id integer not null,
target_id integer not null,
primary key (user_id, target_id)
);

-- Creates `User.name` string field
create function my_schema.users_name(u my_schema.users)
returns varchar as $$
select u.first_name || ' ' || u.last_name;
$$ language sql stable;

-- Creates `User.friends` connection
create function my_schema.users_friends(u my_schema.users)
returns setof my_schema.users as $$
select users.*
from my_schema.users
inner join my_schema.friendships
on (friendships.target_id = users.id)
where friendships.user_id = u.id;
$$ language sql stable;

You can add parameters to your computed column field by declaring additional parameters in your PostgreSQL function:

-- Creates `User.greet(greeting: String!)` string field
create function my_schema.users_greet(
u my_schema.users, --- required table type parameter, unexposed
greeting text --- additional parameter, will be exposed
) returns text as $$
select greeting || ', ' || u.first_name || ' ' || u.last_name || '!';
$$ language sql stable strict;

which can be queried like:

{
userById(id:) {
greet(greeting: "Greetings and salutations")
}
}

Advice

See the advice in the Custom Queries article.