Virtual properties
Properties can be derived from other properties within an object. This lets computed and composite values be created in the object. Such derived properties are named virtual properties. The value of a virtual property can be calculated in two ways:
- 
Using a script called by the onRetrievescript hook. This script then calculates the current value of the virtual property based on the related properties.
- 
Using a query to identify the relationship fields to traverse to reach the managed objects whose state is included in the virtual property, and the fields in these managed objects to include in the value of the virtual property. These properties are called relationship-derived virtual properties (RDVPs). 
Virtual properties using onRetrieve scripts
The onRetrieve script hook lets you run a script when the object is retrieved. In the case of virtual properties, this script gets the data from related properties and uses it to calculate a value for the virtual property. Learn more in Run scripts on managed objects.
| Only run  | 
Relationship-derived virtual properties
Virtual properties can be calculated by IDM based on relationships and relationship notifications. This means that, rather than calculating the current state when retrieved, the managed object that contains the virtual property is notified of changes in a related object, and the virtual property is recalculated when this notification is received. To configure virtual properties to use relationship notifications, there are two areas that need to be configured:
- 
The related managed objects must be configured to use relationship notifications. This lets IDM know where to send notifications of changes in related objects. 
- 
To calculate the value of a virtual property, you must configure which relationships to check, and in which order, a notification of a change in a related object is received. You configure this using the queryConfigproperty.
The queryConfig property tells IDM the sequence of relationship fields it should traverse in order to calculate (or recalculate) a virtual property, and which fields it should return from that related object. This is done using the following fields:
- 
referencedRelationshipFieldsis an array listing a sequence of relationship fields connecting the current object with the related objects you want to calculate the value of the virtual property from. The first field in the array is a relationship field belonging to the same managed object as the virtual property. The second field is a relationship in the managed object referenced by the first field, and so on.For example, the referencedRelationshipFieldsforeffectiveAssignmentsis["roles","assignments"]. The first field refers to therolesrelationship field inmanaged/user, which references themanaged/roleobject. It then refers to theassignmentsrelationship inmanaged/role, which references themanaged/assignmentobject. Changes to either related object (managed/roleormanaged/assignment) will cause the virtual property value to be recalculated, due to thenotify,notifySelf, andnotifyRelationshipsconfigurations on managed user, role, and assignment. These configurations ensure that any changes in the relationships between a user and their roles, or their roles, and their assignments, as well as any relevant changes to the roles or assignments themselves, such as the modification of temporal constraints on roles, or attributes on assignments, will be propagated to connected users, so theireffectiveRolesandeffectiveAssignmentscan be recalculated and potentially synced.
- 
referencedObjectFieldsis an array of object fields that should be returned as part of the virtual property. If this property is not included, the returned properties will be a reference for the related object. To return the entire related object, use*.
- 
flattenPropertiesis a boolean that specifies whether relationship-derived virtual properties should be returned as plain fields rather than as JSON objects with an_idand a_rev. This property isfalseby default.With flattenPropertiesset tofalse, andreferencedObjectFieldsset toname, the response to a query on a user’seffectiveAssignmentsmight look something like this:"effectiveAssignments": [ { "name": "MyFirstAssignment", "_id": "02b166cc-d7ed-46b7-813f-5ed103145e76", "_rev": "2" }, { "name": "MySecondAssignment", "_id": "7162ddd4-591a-413e-a30b-3a5864bee5ec", "_rev": "0" } ]With flattenPropertiesset totrue, andreferencedObjectFieldsset toname, the response to the same query looks like this:"effectiveAssignments": [ "MyFirstAssignment", "MySecondAssignment" ]Setting flattenPropertiestotruealso lets singleton relationship-derived virtual properties be initialized tonull.
Using queryConfig, the virtual property is recalculated when it receives a notice that changes occurred in the related objects. This can be significantly more efficient than recalculating whenever an object is retrieved, while still ensuring the state of the virtual property is correct.
| When you change which fields to return using  | 
The effectiveAssignments property in managed.json is an example of a relationship-derived virtual property:
"effectiveAssignments" : {
    "type" : "array",
    "title" : "Effective Assignments",
    "description" : "Effective Assignments",
    "viewable" : false,
    "returnByDefault" : true,
    "isVirtual" : true,
    "queryConfig" : {
        "referencedRelationshipFields" : [ "roles", "assignments" ],
        "referencedObjectFields" : [ "*" ]
    },
    "usageDescription" : "",
    "isPersonal" : false,
    "items" : {
        "type" : "object",
        "title" : "Effective Assignments Items"
    }
}Create an RDVP
RDVPs are useful because they allow you to query relationships. It’s not possible to query relationship properties directly. For example, you can create an RDVP to query which users have a manager whose email address contains a specific value.
- 
Create a new array property for the managed user object: - 
In the admin UI, go to Configure > Managed Objects > User-type Managed Object and click Add a Property. 
- 
Enter a name for the new RDVP, or optionally, enter a label and select Array as the type. 
- 
Click Save. 
- 
To edit, click the name of the property you created. 
 
- 
- 
On the Details tab, click Show advanced options. 
- 
In the list, enable Virtual and then enable Return by Default. 
- 
Click Save. 
- 
Select the Query Configuration tab. 
- 
In the Referenced Relationship Fields, enter the name of the relationship property or properties used to calculate the RDVP. In this example, the relationship property name is ["manager"]. You must enter the relationship property name as a valid JSON array. 
- 
In Referenced Object Fields, enter the name of the fields to return when the RDVP is calculated. In this example, the field names are _id and mail. 
- 
Click Save. 
Verify the RDVP
After you’ve populated and saved the Referenced Relationship Fields for a user, the RDVP property shows values specified in Referenced Object Fields. For example:
 
These are the RDVP values that you can query. Learn more in Query managed users by RDVP.
Query managed users by RDVP
As mentioned in Create an RDVP, RDVPs are useful because they allow you to query relationship properties that you can’t query directly. For example, if you want to know which organizations have at least one owner or which users have a manager who has certain properties, such as an email that contains a specific value, then you can query an RDVP using REST.
You can query an RDVP using the _queryFilter parameter. For example:
curl \
--header "X-OpenIDM-Username: openidm-admin" \
--header "X-OpenIDM-Password: openidm-admin" \
--header "Accept-API-Version: resource=1.0" \
--request GET \
'http://localhost:8080/openidm/managed/user?_queryFilter=<filter>' (1)| 1 | <filter> is the query expression for the RDVP. Learn more in Construct queries. | 
Example RDVP queries
- 
Which organizations have at least one owner? This query uses the existing ownerIDs RDVP: openidm/managed/organization?_queryFilter=ownerIDS+pr
- 
Which users have a manager whose email address contains the value test?This query uses a custom RDVP where the RDVP is called managerRDVP: _queryFilter=managerRDVP+co+"test"
Enhanced signal propagation
By default, managed objects won’t receive relationship graph topology change signals unless the managed objects have defined a relationship-derived-virtual-property whose state is derived from the graph topology corresponding to the signal.
If graph topology signaling configurations are motivated by requirements other than the correct calculation of relationship-derived virtual properties, you can disable the signal-routing logic in the SignalPropagationCalculator class that is active by default.
To disable the signal-routing logic, you can set disable.signal.propagation.calculator=true in revolver/boot.properties or in system.properties.
| Disabling the routing logic in the  |