DEFINE FIELD statementThe DEFINE FIELD statement allows you to instantiate a named field on a table, enabling you to set the field’s data type, set a default value, apply assertions to protect data consistency, and set permissions specifying what operations can be performed on the field.
DEFINE FIELD statement.DEFINE FIELD statement.
Available since: v3.0.0-alpha.8
NoteIn versions of SurrealDB before 3.0.0-alpha.8,
COMPUTEDfields were implemented using a data type called afuture. Please see the page on futures in this case.
A COMPUTED field is one that is not stored but computed every time it is accessed. Such fields have a more limited set of clauses that can be used. Furthermore, a COMPUTED field cannot be defined on the id field of a record, nor any nested fields (i.e. a field metadata can be defined as computed, but not medatata.can_drive).
The following expression shows the simplest way to use the DEFINE FIELD statement.
-- Declare the name of a field. DEFINE FIELD email ON TABLE user;
The fields of an object and the items in an array can be defined individually using the . operator for objects, or the indexing operator for arrays.
The DEFINE FIELD statement allows you to set the data type of a field. For a full list of supported data types, see Data types.
From version v2.2.0, when defining nested fields, where both the parent and the nested fields have types defined, it is no longer possible to have mismatching types, to prevent any impossible type issues once the schema is defined.
For example, the following will fail:
DEFINE FIELD OVERWRITE fd ON c TYPE { a: string, b: number }; DEFINE FIELD OVERWRITE fd.* ON c TYPE number;
The above will fail with the following error:
Cannot set field `fd[*]` with type `number` as it mismatched with field `fd` with type `{ a: string, b: number } | { a: string, b: bool }`
A | vertical bar can be used to allow a field to be one of a set of types. The following example shows a field that can be a UUID or an int, perhaps for user records that have varying data due to two diffent legacy ID types.
-- Set a field to have either the uuid or int type DEFINE FIELD user_id ON TABLE user TYPE uuid|int;
You can also set a field to have the array data type. The array data type can be used to store a list of values. You can also set the data type of the array’s contents, as well as the maximum number of items that it can hold.
You can make a field optional by wrapping the inner type in an option, which allows you to store NONE values in the field.
The example below shows how to define a field user on a POST table. The field is of type record. This means that the field can store a record<user> or NONE.
DEFINE FIELD user ON TABLE POST TYPE option<record<user>>;
Flexible types allow you to have SCHEMALESS functionality on a SCHEMAFULL table. This is necessary for working with nested object types that need to be able to accept fields that have not yet been defined.
Taking the following CREATE statement:
Without FLEXIBLE, the metadata field will effectively be a SCHEMAFULL object with only a single defined field and thus unable to hold any of the other values that the user attempted to pass in.
With FLEXIBLE, the output will be as expected as the schema now allows any sort of object to be a field on the user table — as long as values for name and metadata.user_id are present.
The same user record without any defined fields or the FLEXIBLE clause would not fail, but none of the data besides the ID for the user record would be recognized.
DEFAULT clause to set a default valueYou can set a default value for a field using the DEFAULT clause. The default value will be used if no value is provided for the field.
DEFAULT and ALWAYS clauseAvailable since: v2.2.0
In addition to the DEFAULT clause, you can use the DEFAULT ALWAYS clause to set a default value for a field. The ALWAYS keyword indicates that the DEFAULT clause is used not only on CREATE, but also on UPDATE if the value is empty (NONE).
With the above definition, the primary field will be set to 123.456 when a new product is created without a value for the primary field or with a value of NONE, and when an existing product is updated if the value is specified the result will be the new value.
In the case of NULL or a mismatching type, an error will be returned.
On the other hand, if a valid number is provided during creation or update, that number will be used instead of the default value. In this case, 123.456.
VALUE clause to set a field’s valueThe VALUE clause differs from DEFAULT in that a default value is calculated if no other is indicated, otherwise accepting the value given in a query.
A VALUE clause, on the other hand, will ignore attempts to set the field to any other value.
As the example above shows, a VALUE clause sets the value every time a record is modified (created or updated). However, the value will not be recalculated in a SELECT statement, which simply accesses the current set value.
To create a field that is calculated each time it is accessed, a computed field can be used.
You can alter a passed value using the VALUE clause. This is useful for altering the value of a field before it is stored in the database.
In the example below, the VALUE clause is used to ensure that the email address is always stored in lowercase characters by using the string::lowercase function.
You can take your field definitions even further by using asserts. Assert can be used to ensure that your data remains consistent. For example you can use asserts to ensure that a field is always a valid email address, or that a number is always positive.
As the ASSERT clause expects an expression that returns a boolean, an assertion with a custom message can be manually created by returning true in one case and using a THROW clause otherwise.
READONLYAvailable since: v1.2.0
The READONLY clause can be used to prevent any updates to a field. This is useful for fields that are automatically updated by the system. To make a field READONLY, add the READONLY clause to the DEFINE FIELD statement. As seen in the example below, the created field is set to READONLY.
DEFINE FIELD created ON resource VALUE time::now() READONLY;
IF NOT EXISTS clauseAvailable since: v1.3.0
The IF NOT EXISTS clause can be used to define a field only if it does not already exist. You should use the IF NOT EXISTS clause when defining a field in SurrealDB if you want to ensure that the field is only created if it does not already exist. If the field already exists, the DEFINE FIELD statement will return an error.
It’s particularly useful when you want to safely attempt to define a field without manually checking its existence first.
On the other hand, you should not use the IF NOT EXISTS clause when you want to ensure that the field definition is updated regardless of whether it already exists. In such cases, you might prefer using the OVERWRITE clause, which allows you to define a field and overwrite an existing one if it already exists, ensuring that the latest version of the definition is always in use
-- Create a field if it does not already exist DEFINE FIELD IF NOT EXISTS email ON TABLE user TYPE string;
OVERWRITE clauseAvailable since: v2.0.0
The OVERWRITE clause can be used to define a field and overwrite an existing one if it already exists. You should use the OVERWRITE clause when you want to modify an existing field definition. If the field already exists, the DEFINE FIELD statement will overwrite the existing definition with the new one.
-- Create an FIELD and overwrite if it already exists DEFINE FIELD OVERWRITE example ON TABLE user TYPE string;
By default, the permissions on a field will be set to FULL unless otherwise specified.
DEFINE FIELD info ON TABLE some_table TYPE string; INFO FOR TABLE some_table;
You can set permissions on fields to control who can perform operations on them using the PERMISSIONS clause. The PERMISSIONS clause can be used to set permissions for SELECT, CREATE, and UPDATE operations. The DELETE operation only relates to records and, as such, is not available for fields.
By using an Access Control List as an example we can show how we can restrict what values can be stored in an array. In this example we are using an array to store the permissions for a user on a resource. The permissions are restricted to a specific set of values.
You can use the ASSERT clause to apply a regular expression to a field to ensure that it matches a specific pattern. In the example below, the ASSERT clause is used to ensure that the countrycode field is always a valid ISO-3166 country code.
While a DEFINE TABLE statement represents a template for any subsequent records to be created, a DEFINE FIELD statement pertains to concrete field data of a record. As such, a DEFINE FIELD statement gives access to the record’s other fields through their names, as well as the current field through the $value parameter.
The $this parameter gives access to the entire record on which a field is defined.
DEFINE FIELD extra_self ON TABLE person VALUE $this; CREATE person:one SET name = "Little person", age = 6;
As DEFINE FIELD statements are computed in alphabetical order, be sure to keep this in mind when using fields that rely on the values of others.
The following example is identical to the above except that full_name has been chosen for the previous field name. The full_name field will be calculated after first_name, but before last_name.
A good rule of thumb is to organize your DEFINE FIELD statements in alphabetical order.
Available since: v2.0.0
A field can also be defined as a literal type, by specifying one or more possible values and/or permitted types.
One more example of a literal containing settings for a full text search filter:
TYPE for the id fieldThe DEFINE FIELD statement can be defined for the id field to specify the acceptable types of IDs, including complex record IDs.
Available since: v2.2.0
A field that is a record link (type record, option<record>, array<record<person>>, and so on) can be defined as a REFERENCE. If this clause is used, any linked to record will be able to define a field of its own of type references which will be aware of the incoming links.
For more information, see the page in the datamodel section on references.