Customizing Schema
Exograph infers the mapping of a type to a table, which is often good enough. However, if you need to customize the mapping, Exograph provides a few annotations. This section will examine the default Exograph mapping and how to customize it.
Type-level customization
For a given type, Exograph infers:
- The associated table name
- The plural version of the type name (used to form query and mutation names).
Exograph sets the default access control rules to disallow anyone executing associated queries and mutations. You can customize this rule using the @access
annotation, but we will defer to the Access Control section.
Table name
By default, Exograph uses the pluralized and snake_cased type name as the table name. For example, Exograph will map the Todo
type to the todos
table, while it will map AuthUser
to the auth_users
table. There is an exception to this rule if a type carries the @plural
annotation, as we'll see next.
You can customize the table's name using the @table
annotation, whose sole single argument specifies the table's name. For example, if you want to map the Todo
to the t_todo
table (typically to match your organization's naming conventions), you can use the @table
annotation as follows:
@table("t_todo")
type Todo {
...
}
While the @table
annotation affects the table name associated with the type, it leaves the query and mutation names unchanged. So, in the above example, the query to get multiple todos will still be called todos
, which is most likely what you want. However, this doesn't work out well in a few cases. For example, consider the following setup:
@table("people")
type Person {
...
}
Exograph will map the Person
to the people
table here. However, the query to get multiple people will still be persons
, which is not ideal. To change the query and mutation names along with the table name, you can use the @plural
annotation.
Table schema
By default, Exograph assumes that a table will be in the public
schema. You can customize its schema by specifying the schema
attribute of the @table
annotation. For example, to map the table User
to be in the auth
schema, you can use the @table
annotation as follows:
@table(schema="auth")
type User {
...
}
The User
type will be mapped to the auth
schema, and the table name will be users
(following the default naming convention). However, if you want to customize the table name as well, you can use the @table
annotation as follows:
@table(schema="auth", name="t_users")
type User {
...
}
The User
type will be mapped to the auth
schema, and the table name will be t_users
.
A common use case is to set the schema of all tables to a specific schema. You can achieve this by setting the schema
attribute of the @postgres
annotation. For example, to set the schema of all tables to entertainment
, you can use the @postgres
annotation as follows:
@postgres(schema="entertainment")
module ConcertModule {
type Concert {
...
venue: Venue
}
type Venue {
...
concerts: Set<Concert>?
}
}
You may still override the schema of a specific table using the @table
annotation and the schema
attribute.
Pluralization
By default, Exograph will use a simple algorithm to pluralize the name of the type. However, it doesn't work well for names with irregular pluralization. For example, Exograph will pluralize person
to persons
, but you will likely want to name it people
. You can control the plural form using the @plural
annotation:
@plural("people")
type Person {
...
}
The @plural
annotation's argument specifies the plural form of the type name. Exograph will use the provided plural form instead of its algorithm when this annotation is present. The plural name affects the table, query, and mutation names. For example, the @plural
annotation above will map the Person
type to the people
table, the query to get multiple people will be people
, and the mutation to delete multiple people will be deletePeople
.
If both @plural
and @table
annotations are present, Exograph will use the argument provided to the @table
annotation to name the table and the argument supplied to the @plural
annotation to name the queries and mutations.
Use the @plural
annotation to deal with type names with irregular pluralization and the @table
annotation to follow your organization's naming conventions.
Using unmanaged tables
Sometimes, you may want to expose a view or a foreign table in your database through Exograph APIs, but not have Exograph manage them for schema migration purposes. For this purpose, you may use the managed=false
attribute of the @table
annotation.
Consider the following view:
CREATE VIEW product_profits AS
SELECT
p.id,
p.name,
p.sale_price,
p.purchase_price,
p.sale_price - p.purchase_price AS profit
FROM products p;
You can define a type for this view as follows:
@postgres
module TodoDatabase {
@access(true)
type Product {
@pk id: Int = autoIncrement()
name: String
salePrice: Float
purchasePrice: Float
}
@table(managed=false)
@access(query=true, mutation=false)
type ProductProfit {
@pk id: Int
name: String
salePrice: Float
purchasePrice: Float
profit: Float
}
}
Exograph will map the ProductProfit
type to the product_profits
view (it could also be a table, possibly a foreign table). However, Exograph will ignore the ProductProfit
type during schema migration.
Exograph will apply access control and infer queries for the ProductProfit
type as usual, including aggregated queries. If you have an unmanaged type representing a view, you will typically want to set mutation=false
, thus removing the mutation APIs for it. However, this is more of a convention than a restriction imposed by Exograph. Therefore, you may put any other access control rules if you want to modify the underlying data through the unmanaged type.
If you allow mutation through a managed type for a view, you will want to make any derived fields read-only. For example, if you allow mutation through the ProductProfit
type, you will want to make the profit
field read-only.
@table(managed=false)
@access(query=true, mutation=true)
type ProductProfit {
@pk id: Int
name: String
salePrice: Float
purchasePrice: Float
@readonly profit: Float
}
An unmanaged type may skip marking a field as @pk
if it doesn't have a primary key. For such a type, Exograph offers only collection and aggregate queries. For example, the ProductProfit
type will have the productProfits
and productProfitsAgg
query but not the productProfit
query (which would take the primary key as an argument).
If you want to mark all types in a module as unmanaged, you can use the @postgres
annotation with the managed=false
attribute.
@postgres(managed=false)
module CommerceViews {
...
}
Here, all the types in the CommerceViews
module will be unmanaged. However, you can override the managed state of a specific type using the @table
annotation.
Field-level customization
Exograph maps each field to a column in the database and infers a few other aspects of the column.
Specifying column name
Exograph infers the column's name as the snake_cased version of the field name for a scalar-type column. For example, Exograph will map the name
field to the name
column, while it will map ticketPrice
to ticket_price
.
You can customize the column's name using the @column
annotation. For example, if you want to map the name
field to the headline
column, you can use the @column
annotation as follows:
type Concert {
...
@column("headline") name: String
}
As discussed earlier, Exograph will infer a relationship between these two types. In the following example, it infers that the foreign key column in the Concert
table is venue_id
. It does so by appending _id
to the field's name (in this case, venue
).
@postgres
module ConcertModule {
type Concert {
...
venue: Venue
}
type Venue {
...
concerts: Set<Concert>?
}
}
If you want to customize the name of the foreign key column, you can use the @column
annotation. For example, to map the venue
field to the venue_pk_
column instead of venue_id
, you can use the following setup:
@postgres
module ConcertModule {
type Concert {
...
@column("venue_pk") venue: Venue
}
type Venue {
...
@column("venue_pk") concerts: Set<Concert>?
}
}
If you change the name of the foreign key column in the Venue
type, you must also change the name of the foreign key column in the Concert
type. This way, the column names guide Exograph to infer the relationship between the two types.
Assigning primary key
The @pk
annotation designates the primary key of a type. The current implementation of Exograph only supports a single primary key (we will lift this restriction in the future):
Auto-incrementing primary key
To use the primary key of integer type, specify the field to be of the Int
type and set its default value of autoIncrement()
. In the following example, the id
field is the primary key, and it will be automatically assigned a value when you create a new concert. Behind the scenes, Exograph will use the SERIAL
type in PostgreSQL by default, but you can customize it to use SMALLSERIAL
or BIGSERIAL
through the @dbtype
annotation, as we will see later.
type Concert {
@pk id: Int = autoIncrement()
...
}
With this arrangement, clients cannot specify the value of the id
field when creating a concert. Exograph will automatically assign a value to the id
field.
Auto-generated Uuid key
To use the primary key of Uuid type, specify the field's type to be Uuid
type with the default value of generate_uuid()
. In the following example, the id
field is the primary key, and it will be automatically assigned a value when you create a new concert. Behind the scenes, Exograph will use the UUID
type in PostgreSQL.
type Concert {
@pk id: Uuid = generate_uuid()
...
}
User-assignable primary key
Auto-generated keys will be the most common form in your model. However, sometimes, you may want to assign a client-specifiable value. For example, you may want to let clients generate a UUID and use it when creating a new entity. You may also need user-assignable primary keys for integration with other systems where you want to sync the primary keys across systems.
To make the primary key user-assignable, skip the default value as follows.
type Venue {
@pk id: Int
...
}
Specifying a default value
The default value of a column is specified using an assignment in the field definition. For example, as we have seen in the previous section, you can set the default value of an Int
field to autoIncrement()
to make it auto-incrementing and the default value of a Uuid
field to generate_uuid()
to make it auto-generated.
Similarly, you can set the default value of a scalar column to a constant value. For example, if you want to set the default value of the price
field to 50
, you can use the following definition:
type Concert {
...
price: Float = 50
}
Setting a default value affects mutations associated with the type. When creating a new concert, the price
field will be optional, and the default value will be used if you don't specify a value.
You can set the default value to now()
for all date and time field types. For example, if you want to set the default value of the createdAt
field to the current time, you can use the following definition:
type Concert {
...
createdAt: Instant = now()
}
When you create a new concert, the createdAt
field will be set to optional, and Exograph will use =the current time if you don't specify a value.
Marking read-only
In the previous example, the intention seems to set the createdAt
field to the creation time. However, this requires clients to not specify a value for the createdAt
field in createConcert
or createConcerts
API. Furthermore, it can be changed to another value when updating the concert (the field remains part of the updateConcert
and updateConcerts
API). Therefore, this arrangement doesn't reflect the intention.
Exograph provides the @readonly
annotation to make a field read-only for such situations. For example, you can make the createdAt
field read-only using the following definition:
type Concert {
...
@readonly createdAt: Instant = now()
}
Now, the createdAt
field will not be part of the createConcert
, createConcerts
, updateConcert
, and updateConcerts
APIs. As a result, the field's value will always be the creation time.
Note that a field marked with the @readonly
annotation must also have a default value.
Updating automatically
What if you want to introduce another field updatedAt
that should be set automatically to the current time whenever the concert is updated? Exograph provides the @update
annotation to achieve this. For example, you can make the updatedAt
field update automatically using the following definition:
type Concert {
...
@update updatedAt: Instant = now()
}
Every time a client updates a concert, Exograph will set the updatedAt
field to its default value, now()
—the current time.
Like the @readonly
annotation, an @update
field must have a default value. The default value usually is a function (like now()
) that generates the value at runtime. It can be constant, but a plain @readonly
is more appropriate. An @update
field is also implicitly considered read-only. Consequently, it will not be part of the createConcert
, createConcerts
, updateConcert
, and updateConcerts
APIs.
You may use the @update
annotation to track other aspects. For example, if you want to set the last modified version, you can use the @update
annotation as follows:
type Concert {
...
@update modificationVersion: Uuid = generate_uuid()
}
Whenever a client updates a concert, Exograph will set the modificationVersion
field to a new UUID.
Controlling Nullability
Exograph will make the column nullable if the field is optional. You can control nullability by adding the ?
suffix to the field type. For example, if you want to make the name
field non-nullable, you can use the following definition:
type TicketPrice {
...
price: Float
details: String?
}
Here, the database schema will have the price
field as non-nullable and the details
field as nullable.
Constraining Uniqueness
Often, you want to set a constraint on a field to make it unique. You may use the @unique
annotation for this purpose. For example, if you want to make sure that the name
field is unique, you can use the @unique
annotation:
type Concert {
...
@unique name: String
}
Here, Exograph will set a database uniqueness constraint on the name
column in the generated schema.
Exograph infers specialized queries to get an entity by unique fields. We will explore this in the queries section.
If you want to mark a specific combination of fields as unique, you can use the @unique
annotation by specifying a name for the unique constraint. For example, if you want to ensure that the combination of emailId
and emailDomain
is unique, you can use the @unique
annotation specifying a name for the unique constraint:
type Person {
...
@unique("email") emailId: String
@unique("email") emailDomain: String
}
You can pass an array of field names to the @unique
annotation to specify a unique constraint. For example, if you want to make sure that the combination of primaryEmailId
and emailDomain
as well as the combination secondaryEmailId
and emailDomain
is unique, you can use the @unique
annotation specifying names for the unique constraint:
type Person {
@unique("primary_email") primaryEmailId: String
@unique("secondary_email") secondaryEmailId: String?
@unique("primary_email", "secondary_email") emailDomain: String
}
Since we use the name primary_email
for the primaryEmailId
and the emailDomain
fields, that combination will be marked unique. We do the same for the secondaryEmailId
and the emailDomain
fields.
Adding Index
It is a common practice to set up indexes on columns to speed up queries. While indexes speed up queries, they slow down inserts and updates. So, you should analyze the usage pattern of your application and create indexes accordingly.
By default, Exograph will not set up any explicit indexes. However, note that Postgres will set up indices for primary key columns and those with a uniqueness constraint (see @unique
above). Exograph offers the @index
annotation to allow you to set up appropriate indices. The @index
annotation follows the same syntax as the @unique
annotation. For example, if you want to create an index on the age
column, you can use the @index
annotation as follows:
type Person {
...
@index age: Int
}
Here, Exograph will create an index named person_age_idx
on the age
column. If you want to control the name of the index, you can use the @index
annotation as follows:
type Person {
...
@index("person_age_index") age: Int
}
Suppose the application's usage pattern suggests that you must create an index on a combination of fields (typically, a frequent query that supplies a where
clause with multiple fields). You can use the @index
annotation by specifying a name for the index. For example, if you want to create an index on the combination of firstName
and lastName
, you can use the @index
annotation specifying a name for the index:
type Person {
...
@index("person_name") firstName: String
@index("person_name") lastName: String
}
Here, Exograph will create an index with the name person_name
on the columns for the firstName
and lastName
fields.
Like the @unique
annotation, you can pass an array of field names to the @index
annotation to specify an index. For example, suppose you need to create an index on the combination of firstName
and lastName
and those fields individually. You can use the @index
annotation specifying names for the index:
type Person {
...
@index("person_name", "person_first_name") firstName: String
@index("person_name", "person_last_name") lastName: String
}
Here, Exograph will set up three indices: one on the firstName
field, one on the lastName
field, and one on the combination of the firstName
and lastName
fields.
Customizing field type
Exograph infers the column type based on the field type. For example, if the field type is String
, the column type will be inferred as TEXT
. However, you may want more precise control over the database column type. Exograph offers a few annotations for this purpose.
Explicit column type
If you need to control the mapping of a database column type directly, you can use the @dbtype
annotation. For example, if you want to set the column type of the name
field to VARCHAR(100)
, you can use the @type
annotation:
@dbtype("VARCHAR(100)") name: String
The @dbtype
annotation takes a string argument: the column type. You can use any valid PostgreSQL type compatible with the field type. For example, if you want to set the column type of the price
field to SMALLINT
, you can use the @dbtype
annotation as follows:
@dbtype("SMALLINT") price: Int
Ensure that the type you specify is compatible with the field type. For example, if you specify the type as VARCHAR(100)
for a field of type Int
, the generated schema will be invalid.
The @dbtype
annotation offers low-level control over the column type. Exograph also provides a few type-specific annotations. These annotations indirectly have the same effect as using the @dbtype
annotation but offer a more convenient way to customize the column type.
Specifying both the @dbtype
and the type-specific annotations is an error.
String field type
By default, the column type will allow the string to be of any length (to the extent the database allows it). However, you can use the @maxLength
annotation to restrict it. For example, if you want to limit the length of the name
field to 100 characters, you can use the @maxLength
annotation:
@maxLength(100) description: String
Here, the column type will be set to VARCHAR(100)
(instead of TEXT
).
Integer field type
Exograph offers a few annotations for integer fields to customize the column type. They all have the effect of setting the column type to one of SMALLINT
, INTEGER
, or BIGINT
in addition to setting constraints on the value of the field. The annotations are:
@bits16
,@bits32
, andbits64
: These annotations specify the number of bits in the integer. For example, if you want to set the column type of themask
field to contain only 16 bits, you can use the@bits*
annotation as follows:
@bits16 mask: Int
You can specify only one of the @bits*
annotations.
@range
: This annotation specifies the range of the integer. It takes two arguments: the minimum and the maximum value. For example, if you want to set the column type of theage
field to 0 to 200, you can use the@range
annotation:
@range(min = 0, max = 200) age: Int
You may use the @range
annotation with one of the @bits*
annotations. Exograph will infer the column type based on the range and integer size.
Float field type
Exograph offers the @singlePrecision
and @doublePrecision
annotations for float fields to customize the column type. They all have the effect of setting the column type to REAL
or DOUBLE PRECISION
.
Decimal field type
Floating point numbers are OK if you can tolerate some loss of precision. However, you can use the Decimal
type to store the exact decimal numbers. For example, if you want to store the price of a concert ticket, you can use the Decimal
type. For this type, Exograph offers the following annotations to customize the column type:
-
@precision
: This annotation specifies the total number of digits in the decimal number. -
@scale
: This annotation specifies the number of digits after the decimal point. For example, if you want to set the column type of theprice
field to 2 digits after the decimal point, you can use the@scale
annotation:
@precision(5) @scale(2) price: Decimal
The above will set the column type to NUMERIC(5, 2)
and allow numeric values between 0.00 and 999.99. See the PostgreSQL documentation for more details.
If you specify @scale
, you must also specify @precision
.
Date and Time field type
For date and time fields (LocalDateTime
, LocalDate
, and Instance
), Exograph offers the @precision
, which then maps it to Postgres's precision. See the PostgreSQL documentation for more details.