Adding Field Validations
Liferay DXP 2023.Q3+/Portal GA92+
Validations set rules for determining valid field entries. Each validation has its own trigger, conditions, and error message, which you can set via the Objects UI. You can define validations using Groovy scripts or Liferay expressions.
A third type of validation is available as of Liferay DXP 2024.Q1+/GA112+: Composite Key validation. Use composite key validations to require that a combination of two or more fields be unique.

- If out-of-the-box validations don’t meet your needs, you can create custom validations using the objectValidationRuleclient extension. See Microservice Client Extensions for more information.
- Validation messages for system objects may not be displayed in their native UI. However, the validations run as expected.
To add a validation,
- 
Open the Global Menu (  ), go to the Control Panel tab, and click Objects. ), go to the Control Panel tab, and click Objects.
- 
Begin editing an object definition. 
- 
Go to the Validations tab and click Add (  ). ).
- 
Enter a label and select a validation type: Groovy, Expression Builder, or Composite Key. If you’ve deployed object validation rule client extensions to your instance, they also appear as options.  
- 
Click Save. 
- 
Begin editing the validation. 
- 
In the Basic Info tab, activate the validation.  
- 
Select the trigger event to determine when the validation runs. Each validation can only have one trigger event. 
- 
Add validation rules. For Groovy and Expression Builder validations, go to the Conditions tab and add conditions to the validation. Conditions can include multiple fields and functions for performing complex logic.  When using Groovy, you can browse and add available data fields to your conditions via the side panel. See Using Groovy Validations for more information. When using Expression Builder, you can browse and add fields, operators, and functions to your conditions via the side panel. See Using Expression Builder Validations for more information. Important- Groovy script validations are only available for Liferay PaaS and Liferay DXP Self-Hosted.
- As of DXP 2024.Q3, scripting is disabled by default. You can enable it in System Settings → Script Management (under the Security category).
 To set up composite key validations, go to the Unique Composite Key tab and select the fields to use as a composite key. See Using Composite Key Validations for more information.  
- 
Enter a localizable error message. This message appears whenever the validation is triggered, and field entries do not meet one or more of the defined conditions. 
- 
Select an output validation type to determine where the error message appears. Full Validation (Form Summary): Display the error message at the top of the form. Partial Validation (Inline Field): Display the error message next to the specified field. This feature does not work in object layouts.  
- 
Click Save. 
While activated, the validation runs for all new object entries and is displayed in layouts and form containers.

Using Groovy Validations
Liferay PaaS and Liferay DXP Self-Hosted
As of DXP 2024.Q3, scripting is disabled by default. You can enable it in System Settings → Script Management (under the Security category).
Groovy validations support all standard Groovy Script capabilities. When defining conditions, you must use the invalidFields variable. Liferay only displays the validation error message when invalidFields returns true.

Liferay uses the GroovyShell class to check your Groovy scripts for valid syntax when you click Save. If the script is invalid, Liferay shows an error message.
Using Expression Builder Validations
Expression Builder provides predefined fields, operators, and functions that you can access in the Elements side panel. Clicking an element adds it to the conditions editor. These functions return a Boolean value. See Expression Builder Validations Reference for a complete list of provided operators and functions.
You can only use Expression Builder validations with text, numeric, date, and Boolean field types.

Liferay checks your expression for valid syntax when you click Save. If the expression is invalid, Liferay shows an error message.
Expression Builder Operators
This table lists available operators for Expression Builder validations:
| Operator | Description | 
|---|---|
| And ( AND) | Coordinating conjunction used to indicate a dependent relationship | 
| Divided By ( /) | Mathematical operator for division | 
| Minus ( -) | Mathematical operator for subtraction | 
| Or ( OR) | Coordinating conjunction used to indicate an independent relationship | 
| Plus ( +) | Mathematical operator for addition | 
| Multiply ( *) | Mathematical operator for multiplication | 
Expression Builder Functions
This table lists available Expression Builder functions with their compatible field types:
| Operator | Text Fields | Numeric Fields | Date Fields | Description | 
|---|---|---|---|---|
| Compare Dates | ✔ | Checks if a date field’s value is the same as a set value. | ||
| Concat | ✔ | Combines multiple strings or text fields and returns a single string that you can use with other validation functions. | ||
| Condition | ✔ | ✔ | ✔ | Checks if user input meets one or more conditions and returns a Boolean value. | 
| Contains | ✔ | ✔ | Checks if a field contains a specified value and returns a Boolean. | |
| Does Not Contain | ✔ | ✔ | Checks if a field does not contain a specified value and returns a Boolean. | |
| Future Dates | ✔ | Checks if a date field’s value is in the future and returns a Boolean. | ||
| Is a URL | ✔ | Checks if a text field is a URL and returns a Boolean. | ||
| Is an Email | ✔ | Checks if a text field is an email and returns a Boolean. | ||
| Is Decimal | ✔ | Checks if a number field is a decimal and returns a Boolean. | ||
| Is Empty | ✔ | Checks if a text field is empty and returns a Boolean. | ||
| Is Equal To | ✔ | ✔ | Checks if a field value is equal to a specified value and returns a Boolean. | |
| Is Greater Than | ✔ | Checks if a number field is greater than a specific numeric value and returns a Boolean. | ||
| Is Greater Than or Equal To | ✔ | Checks if a number field is greater than or equal to a specific numeric value and returns a Boolean. | ||
| Is Integer | ✔ | Checks if a number field is an integer and returns a Boolean. | ||
| Is Less Than | ✔ | Checks if a number field is less than a specific numeric value and returns a Boolean. | ||
| Is Less Than or Equal To | ✔ | Checks if a number field is less than or equal to a specific numeric value and returns a Boolean. | ||
| Is Not Equal To | ✔ | ✔ | Checks if a field value is different from a specified value and returns a Boolean. | |
| Match | ✔ | Checks if a text field matches a specific string value or RegEx expression and returns a Boolean. | ||
| Old Value | ✔ | ✔ | ✔ | Retrieves the prior value for the specified field. | 
| Past Dates | ✔ | Checks if a date field’s value is in the past and returns a Boolean. | ||
| Range | ✔ | Checks if a date range begins with a past date and ends with a future date and returns a Boolean. | ||
| Sum | ✔ | Adds multiple numeric fields together and returns a single number that you can use with other validation functions. | 
See Expression Builder Validations Reference for more information and examples.
Available Fields Reference
When constructing conditions, you can use any of the object’s custom or system fields. From Liferay DXP 2025.Q2+, you can also select from relationship fields on the child side of a one-to-many relationship.
Below are all default fields available for custom objects:
| Field | Description | 
|---|---|
| companyId | Portal instance where the entry was created | 
| createDate | When the entry was created | 
| currentDate | Date when the entry is submitted | 
| currentUserId | ID of the user submitting the entry | 
| externalReferenceCode | External reference code for the entry | 
| groupId | Site ID in where the entry was created | 
| lastPublishDate | Date when the entry was last published | 
| modifiedDate | Date when the entry was last modified | 
| mvccVersion | MVCC version of the entry | 
| objectDefinitionId | ID of the entry’s object | 
| objectEntryId | ID for the entry | 
| status | Workflow status for the entry | 
| statusByUserId | ID of the assigned user in Workflow | 
| statusByUserName | Name of the assigned user in Workflow | 
| statusDate | Date when the Workflow status was last updated | 
| userId | ID of the entry’s author | 
| userName | User name of the entry’s author | 
| uuid | Unique universal ID for the entry | 
System objects have their own default fields, though there is some overlap with the above chart.
Using Composite Key Validations
Liferay DXP 2024.Q1+/GA112+ (Release Feature Flag)
Liferay DXP 2024.Q3+/Portal GA125+ (Generally Available)
If your version of Liferay requires it, first enable the release feature flag for composite key validation. Go to Global Menu ( ) → Control Panel → Instance Settings → Feature Flags. Open the Release section and enable Improve Field Validations (LPS-187854).
) → Control Panel → Instance Settings → Feature Flags. Open the Release section and enable Improve Field Validations (LPS-187854).
Some data models and applications require unique composite keys. For example, an orders object can require that each combination of customer ID and order date are unique in the system. Once enforced, this composite key can be used to look up and work with unique order entries. The Unique Composite Key validation ensures that a combination of fields is unique in the scope of the object. Site-scoped objects can have unique composite keys in the site, while instance-scoped objects can have unique composite keys throughout the instance.
You can use text, integer, and picklist fields in composite keys. You can also use relationship fields on the child side of a one-to-many relationship.

You cannot add a field to a composite key if it already has data.

Certain actions are only supported in draft objects:
| Action | Object Status | Supported | 
|---|---|---|
| Delete the validation | Draft Published | ✔ ✔ | 
| Remove a field from the validation | Draft Published | ✔ ✘ | 
| Delete a field that’s being used in a validation | Draft Published | ✘ ✘ | 
Executing Validation Rules with APIs
Liferay DXP 2025.Q2+
There’s one out-of-the-box trigger event: On Submission. If this doesn’t satisfy your use case, consider triggering the validation with your custom object’s validation API endpoint. For example, in a multi-step form you might want to validate each page before transitioning to the next page. Users with permission to POST an object entry can also call its validation endpoint.
To use the validation endpoint, pass the external reference codes (ERCs) of the validation rules to trigger in the request body; passing no validation rule ERCs triggers all of them. Additionally, pass the field names and values to validate in the request body. In this example, two dates are passed with a single validation rule ERC. This validation rule ensures that the startDate field value is always earlier than the endDate value:
curl \
   "http://localhost:8080/o/c/timeoffrequests/validate" \
   --data-raw '
      {
         "objectValidationRuleExternalReferenceCodes": ["{validationRuleERC}"],
         "values": {
            "endDate": "2025-06-01",
            "startDate": "2025-06-08"
         }
      }' \
   --header "Content-Type: application/json" \
   --request "POST" \
   --user "test@liferay.com:learn"
Replace the {validationRuleERC} with the ERC of a validation rule in the object definition.
To find the validation rule ERCs for an object definition, use the object-admin API. For example, if the ERC of the object definition is C_TIMEOFFREQUEST, you can execute this GET request:
curl \
   "http://localhost:8080/o/object-admin/v1.0/object-definitions/by-external-reference-code/C_TIMEOFFREQUEST/object-validation-rules" \
   --user "test@liferay.com:learn"
Basic authentication is used for demonstration purposes. For production, you should authorize users via OAuth2. See Using OAuth2 to Authorize Users for a sample React application that uses OAuth2.