Skip to content

Latest commit

 

History

History
1365 lines (1252 loc) · 60.6 KB

centralized-provisioning-with-identity-directory-9d0235c.md

File metadata and controls

1365 lines (1252 loc) · 60.6 KB

Centralized Provisioning with Identity Directory

Identity Directory is the persistency layer of SAP Cloud Identity Services, providing a central place for storing and managing users and groups. You can use it in centralized provisioning scenarios for managing user access to SAP cloud applications from a single, central location.

In a typical scenario, illustrated in the diagram below, the Identity Provisioning service reads identities from the customer's user store, like Microsoft Entra ID, and replicates them to the directory. From there, they are provisioned further to various SAP cloud systems like SAP S/4HANA Cloud, SAP Analytics Cloud and more.

For every newly created user, the directory generates a Global User ID, which serves as unique identifier across the entire landscape. This unique correlation attribute is then distributed by Identity Provisioning to the SAP cloud applications that need it for their integration with the SAP Task Center, for example. The unique identifier can also be generated externally and persisted in the identity directory during the initial replication from the customer’s user store. For more information, see Global User ID in Integration Scenarios.

You can access the Identity Directory in the following ways:

  • Use the Identity Provisioning service to create, update and delete users and groups by running provisioning jobs. The service provides a connector, known as Local Identity Directory, which can be set up as the source, target or proxy system for provisioning scenarios. There is no need to configure connectivity and authentication properties when connecting the Identity Directory and Identity Provisioning. Both services share the same SAP Cloud Identity Services tenant that you are currently logged in.

    For more information, see Local Identity Directory.

    Note:

    The Local Identity Directory connector is available for bundle and standalone tenants running on SAP Cloud Identity Services infrastructure.

  • Use the SCIM 2.0 API to programmatically access and manage identities in the directory. The API allows you to view predefined schemas and to define custom schemas with your own attributes. For more information, see Identity Directory API.

  • Use the SAP Cloud Identity Services administration console to access and manage identities in the directory via the user interface. For more information, see Managing Users and Managing Groups.

Configure Local Identity Directory, both as a target and as a source system.

In this scenario, the Local Identity Directory is initially set up as a target system for user and group provisioning. Typically, identities are initially provisioned from the customer's user store to the directory, thus establishing the user store as the leading source system for identity management. Following this, users and groups from other source systems are provisioned to the directory. Finally, the identities in the directory are replicated to the target systems.

The following configuration example uses a number of systems, including: Microsoft Entra ID, Local Identity Directory, and SAP S/4HANA Cloud.

  1. Sign in to SAP Cloud Identity Services administration console and navigate to Identity Provisioning.

  2. Add Microsoft Entra ID as a source system.

  3. Add Local Identity Directory as a target system.

  4. Optional: Configure properties that are relevant for the Local Identity Directory target system, for example idds.support.bulk.operation and idds.bulk.operations.max.count. For more information, see List of Properties.

  5. Optional: Review and adapt the default write transformations as necessary.

    Code Syntax:

    {
      "user": {
        "condition": "($.emails EMPTY false) && ($.userName EMPTY false) && isValidEmail($.emails[0].value)",
        "mappings": [
          {
            "sourceVariable": "entityIdTargetSystem",
            "targetPath": "$.id"
          },
          {
            "constant": ["urn:ietf:params:scim:schemas:core:2.0:User","urn:ietf:params:scim:schemas:extension:enterprise:2.0:User","urn:ietf:params:scim:schemas:extension:sap:2.0:User","urn:sap:cloud:scim:schemas:extension:custom:2.0:User"],
            "targetPath": "$.schemas"
          },
          {
            "sourcePath": "$.userName",
            "targetPath": "$.userName"
          },
          {
            "sourcePath": "$.emails",
            "preserveArrayWithSingleElement": true,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails']",
            "scope": "createEntity",
            "functions": [
               {
                     "function": "putIfAbsent",
                     "key": "verified",
                     "defaultValue": true
               }
             ]
          },
          {
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails'][*]['type']",
            "type": "remove"
          },
          {
            "sourcePath": "$.emails[*].value",
            "preserveArrayWithSingleElement": true,
            "targetPath": "$.emails[?(@.value)]"
          },
          {
            "sourcePath": "$.name.givenName",
            "targetPath": "$.name.givenName",
            "optional": true
          },
          {
            "sourcePath": "$.name.middleName",
            "targetPath": "$.name.middleName",
            "optional": true
          },
          {
            "sourcePath": "$.name.familyName",
            "targetPath": "$.name.familyName",
            "optional": true
          },
          {
            "sourcePath": "$.name.honorificPrefix",
            "targetPath": "$.name.honorificPrefix",
            "optional": true
          },
          {
            "sourcePath": "$.addresses",
            "targetPath": "$.addresses",
            "preserveArrayWithSingleElement": true,
            "defaultValue": [],
            "optional": true,
            "functions": [
              {
                "function": "putIfAbsent",
                "key": "type",
                "defaultValue": "work"
              },
              {
                "condition": "(@.type NIN ['work', 'home'])",
                "function": "putIfPresent",
                "key": "type",
                "defaultValue": "work"
              }
            ]
          },
          {
            "sourcePath": "$.phoneNumbers",
            "targetPath": "$.phoneNumbers",
            "preserveArrayWithSingleElement": true,
            "optional": true
          },
          {
            "sourcePath": "$.displayName",
            "targetPath": "$.displayName",
            "optional": true
          },
          {
            "sourcePath": "$.userType",
            "targetPath": "$.userType",
            "defaultValue": "employee",
            "optional": true
          },
          {
            "sourcePath": "$.locale",
            "targetPath": "$.locale",
            "optional": true
          },
          {
            "sourcePath": "$.timezone",
            "targetPath": "$.timezone",
            "optional": true
          },
          {
            "sourcePath": "$.active",
            "targetPath": "$.active",
            "defaultValue": true,
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validFrom']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validFrom']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validTo']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validTo']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
            "optional" : true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
            "optional" : true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
            "optional" : true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
            "optional" : true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
            "optional" : true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
            "optional" : true,
            "functions": [
                {
                  "function": "resolveEntityIds"
                }
              ]
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
            "optional" : true
          },
          {
            "constant": false,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sendMail']",
            "scope": "createEntity"
          },
          {
            "constant": true,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['mailVerified']",
            "scope": "createEntity"
          },
          {
            "constant": "disabled",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['passwordDetails']['status']",
            "scope": "createEntity"
          },
          {
            "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']['attributes']",
            "preserveArrayWithSingleElement": true,
            "optional": true,
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']['attributes']"
          },
          {
            "constant": "<your-initial-password>",
            "targetPath": "$.password",
            "scope": "createEntity",
            "ignore": true
          },
          {
            "constant": "<your-source-system-type-code>",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sourceSystem']",
            "scope": "createEntity",
            "ignore": true
          },
          {
            "constant": "<your-source-system-id>",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sourceSystemId']",
            "scope": "createEntity",
            "ignore": true
          }
        ]
      },
      "group": {
        "mappings": [
          {
            "sourceVariable": "entityIdTargetSystem",
            "targetPath": "$.id"
          },
          {
            "constant": ["urn:ietf:params:scim:schemas:core:2.0:Group","urn:sap:cloud:scim:schemas:extension:custom:2.0:Group","urn:ietf:params:scim:schemas:extension:sap:2.0:Group"],
            "targetPath": "$.schemas"
          },
          {
            "sourcePath": "$.displayName",
            "targetPath": "$.displayName"
          },
          {
            "sourcePath": "$.members[*].value",
            "preserveArrayWithSingleElement": true,
            "optional": true,
            "targetPath": "$.members[?(@.value)]",
            "functions": [
                {
                  "entityType": "user",
                  "type": "resolveEntityIds"
                }
              ]
          },
          {
            "sourcePath": "$.members[*].value",
            "preserveArrayWithSingleElement": true,
            "optional": true,
            "targetPath": "$.members[?(@.value)]",
            "functions": [
                {
                  "entityType": "group",
                  "type": "resolveEntityIds"
                }
              ]
          },
          {
            "sourcePath": "$.displayName",
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
            "scope": "createEntity"
          },
          {
            "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
            "optional": true,
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
            "scope": "createEntity"
          },
          {
            "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['description']",
            "optional": true,
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['description']"
          },
          {
            "condition": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId'] EMPTY false",
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId']"
          },
          {
            "condition": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId'] EMPTY false",
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']",
            "optional": true,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']"
          },
          {
            "condition": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId'] EMPTY false",
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']",
            "optional": true,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']"
          }
        ]
      }
    }
    
  6. Schedule or run manually a provisioning job from the Microsoft Entra ID source system.

  7. View the provisioning job logs and subscribe to receive job notifications. We recommend that you do this for each source system where a provisioning job is initiated.

  8. Add Local Identity Directory as a source system.

    It already contains all the users and groups provisioned from Microsoft Entra ID.

  9. Optional: Configure properties that are relevant for the Local Identity Directory source system, for example idds.user.filter and idds.group.filter. For more information, see List of Properties.

  10. Optional: Review and adapt the default read transformations as necessary.

    Code Syntax:

    {
        "user": {
            "mappings": [
                {
                    "sourcePath": "$.id",
                    "targetVariable": "entityIdSourceSystem",
                    "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']['userId']"
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['userId']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['userId']"
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['userUuid']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['userUuid']"
                },
                {
                    "sourcePath": "$.schemas",
                    "preserveArrayWithSingleElement": true,
                    "targetPath": "$.schemas"
                },
                {
                    "sourcePath": "$.userName",
                    "targetPath": "$.userName",
                    "optional": true,
                    "correlationAttribute": true
                },
                {
                    "sourcePath": "$.name.givenName",
                    "targetPath": "$.name.givenName",
                    "optional": true
                },
                {
                    "sourcePath": "$.name.middleName",
                    "targetPath": "$.name.middleName",
                    "optional": true
                },
                {
                    "sourcePath": "$.name.familyName",
                    "targetPath": "$.name.familyName",
                    "optional": true
                },
                {
                    "sourcePath": "$.name.honorificPrefix",
                    "targetPath": "$.name.honorificPrefix",
                    "optional": true
                },
                {
                    "sourcePath": "$.emails[*].value",
                    "preserveArrayWithSingleElement": true,
                    "targetPath": "$.emails[?(@.value)]"
                },
                {
                    "sourcePath": "$.emails[?(@.primary== true)].value",
                    "correlationAttribute": true
                },
                {
                    "sourcePath": "$.active",
                    "targetPath": "$.active"
                },
                {
                    "sourcePath": "$.userType",
                    "targetPath": "$.userType",
                    "optional": true
                },
                {
                    "sourcePath": "$.addresses",
                    "targetPath": "$.addresses",
                    "preserveArrayWithSingleElement": true,
                    "optional": true
                },
                {
                    "sourcePath": "$.locale",
                    "targetPath": "$.locale",
                    "optional": true
                },
                {
                    "sourcePath": "$.phoneNumbers",
                    "targetPath": "$.phoneNumbers",
                    "preserveArrayWithSingleElement": true,
                    "optional": true
                },
                {
                    "sourcePath": "$.timezone",
                    "targetPath": "$.timezone",
                    "optional": true
                },
                {
                    "sourcePath": "$.displayName",
                    "targetPath": "$.displayName",
                    "optional": true
                },
                {
                    "sourcePath": "$.groups",
                    "targetPath": "$.groups",
                    "preserveArrayWithSingleElement": true,
                    "optional": true
                },
                {
                    "condition": "$.displayName EMPTY true",
                    "type": "remove",
                    "targetPath": "$.displayName"
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validFrom']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validFrom']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validTo']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validTo']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sourceSystem']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sourceSystem']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sourceSystemId']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sourceSystemId']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']",
                    "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']",
                    "optional": true
                },
                {
                    "sourcePath": "$.company",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
                    "optional": true
                }
            ]
        },
        "group": {
            "mappings": [
                {
                    "sourcePath": "$.id",
                    "targetVariable": "entityIdSourceSystem"
                },
                {
                    "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
                    "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']"
                },
                {
                    "sourcePath": "$.displayName",
                    "targetPath": "$.displayName"
                },
                {
                    "sourcePath": "$.members",
                    "targetPath": "$.members",
                    "preserveArrayWithSingleElement": true,
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']",
                    "optional": true
                },
                {
                    "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']",
                    "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']",
                    "optional": true
                }
            ]
        }
    }
    
  11. Add SAP S/4HANA Cloud as a target system.

  12. Start another provisioning job, this time from the Local Identity Directory source system.

Results

The Microsoft Entra ID users and groups are provisioned to the identity directory. It now becomes the central point of truth for identity information. Once in the directory, the users and groups are replicated to the SAP S/4HANA Cloud target.

The personal and technical information of the user varies across different systems. To manage a unified and up-to-date user data, you can provision the identities from multiple sources and merge them into one target identity directory.

You have wide flexibility to choose your own approach when merging data from various sources. This could include defining the unique attribute or a combination of them when resolving conflicts, deciding on how to add and update user and group attributes. All of these require changes in the identity directory configuration.

Caution:

When reading user data from multiple sources, we strongly recommend that you perform consecutive provisioning jobs. Simultaneous jobs may lead to inconsistent or overwritten user data in the target system.

Note:

To execute PATCH instead of PUT requests when updating an entry, set the idds.support.patch.operation = true on the Properties tab of the Local Identity Directory target system.

Merge Users by Email as Correlation Attribute

By default, Identity Provisioning determines the user uniqueness based on the userName attribute. This means that if a user with a particular userName already exists in the identity directory, the service will match any new user, provisioned from other source systems with the same userName, to this existing user and update it accordingly.

To merge the user data by email instead of userName, you only need to set the idds.user.unique.attribute to emails[0].value. For more information about the identity directory unique attributes supported for conflict resolution, see List of Propertiesidds.user.unique.attribute.

Merge User Attributes

Following the provisioning of your identities from the leading source system, you may want to replicate additional user attributes from the second source system which were not available in the first one.

To achieve this, we suggest that you configure two pairs of source-target systems. Each target system should point to the same identity directory instance, although both will have different configurations. For example, if Microsoft Entra ID (your leading source system) and Local Identity Directory 1 (target) make up your first pair, then the second pair could consist of SAP SuccessFactors (source) and Local Identity Directory 2 (target). This setup allows you to control which user attribute comes from which source system. The overall process is the following:

  1. Create the first pair of source-target systems: Microsoft Entra ID > Local Identity Directory 1.

  2. On the Properties tab of the target system set the idds.support.patch.operation = true and the idds.user.unique.attribute to a unique attribute of your choice.

    Note:

    The unique attribute you set for the idds.user.unique.attribute property can either be the same or different in each of your target systems. However, it must be unique in each case. For example, if userName and email are unique attributes, you can set userName in the Local Identity Directory 1 and emails[0].value in the Local Identity Directory 2.

  3. Modify the write transformation of the target system to provision particular user attributes. For example, keep the required id, userName and emails, as well as the givenName, familyName, displayName and addresses.

    Code Syntax:

      {
       "user":{
          "condition":"($.emails EMPTY false) && ($.userName EMPTY false) && isValidEmail($.emails[0].value)",
          "mappings":[
             {
                "sourceVariable":"entityIdTargetSystem",
                "targetPath":"$.id"
             },
             {
                "constant":[
                   "urn:ietf:params:scim:schemas:core:2.0:User",
                   "urn:ietf:params:scim:schemas:extension:sap:2.0:User",
                   "urn:sap:cloud:scim:schemas:extension:custom:2.0:User"
                ],
                "targetPath":"$.schemas"
             },
             {
                "sourcePath":"$.userName",
                "targetPath":"$.userName"
             },
             {
                "sourcePath":"$.emails",
                "preserveArrayWithSingleElement":true,
                "targetPath":"$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails']",
                "scope":"createEntity",
                "functions":[
                   {
                      "function":"putIfAbsent",
                      "key":"verified",
                      "defaultValue":true
                   }
                ]
             },
             {
                "targetPath":"$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails'][*]['type']",
                "type":"remove"
             },
             {
                "sourcePath":"$.emails[*].value",
                "preserveArrayWithSingleElement":true,
                "targetPath":"$.emails[?(@.value)]"
             },
             {
                "sourcePath":"$.name.givenName",
                "targetPath":"$.name.givenName",
                "optional":true
             },
             {
                "sourcePath":"$.name.familyName",
                "targetPath":"$.name.familyName",
                "optional":true
             },
             {
                "sourcePath":"$.addresses",
                "targetPath":"$.addresses",
                "preserveArrayWithSingleElement":true,
                "defaultValue":[
                   
                ],
                "optional":true,
                "functions":[
                   {
                      "function":"putIfAbsent",
                      "key":"type",
                      "defaultValue":"work"
                   },
                   {
                      "condition":"(@.type NIN ['work', 'home'])",
                      "function":"putIfPresent",
                      "key":"type",
                      "defaultValue":"work"
                   }
                ]
             },
             {
                "sourcePath":"$.displayName",
                "targetPath":"$.displayName",
                "optional":true
             },
             {
                "constant":true,
                "targetPath":"$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['mailVerified']",
                "scope":"createEntity"
             }
          ]
       },
       "group":{
          "mappings":[
     
     .....
    
  4. Run the provisioning job from Microsfot Entra ID and replicate the identities to the identity directory instance.

  5. Create the second pair of source-target systems: SAP SuccessFactors > Local Identity Directory 2.

  6. Set the idds.support.patch.operation = true and the idds.user.unique.attribute to a unique attribute of your choice.

  7. Modify the write transformation of the target system to provision another set of user attributes. For example, keep the required id, userName and emails, as well as the userType, department and division and map attributes from the SAP SuccessFactors extension schema perPersonUuid and personIdExternal to custom attributes.

    Code Syntax:

    {
       "user":{
          "condition":"($.emails EMPTY false) && ($.userName EMPTY false) && isValidEmail($.emails[0].value)",
          "mappings":[
             {
                "sourceVariable":"entityIdTargetSystem",
                "targetPath":"$.id"
             },
             {
                "constant":[
                   "urn:ietf:params:scim:schemas:core:2.0:User",
                   "urn:ietf:params:scim:schemas:extension:enterprise:2.0:User",
                   "urn:ietf:params:scim:schemas:extension:sap:2.0:User",
                   "urn:sap:cloud:scim:schemas:extension:custom:2.0:User"
                ],
                "targetPath":"$.schemas"
             },
             {
                "sourcePath":"$.userName",
                "targetPath":"$.userName"
             },
             {
                "sourcePath":"$.emails",
                "preserveArrayWithSingleElement":true,
                "targetPath":"$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails']",
                "scope":"createEntity",
                "functions":[
                   {
                      "function":"putIfAbsent",
                      "key":"verified",
                      "defaultValue":true
                   }
                ]
             },
             {
                "targetPath":"$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails'][*]['type']",
                "type":"remove"
             },
             {
                "sourcePath":"$.emails[*].value",
                "preserveArrayWithSingleElement":true,
                "targetPath":"$.emails[?(@.value)]"
             },
             {
                "sourcePath":"$.userType",
                "targetPath":"$.userType",
                "defaultValue": "employee",
                "optional":true
             },
             {
                "sourcePath":"$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
                "targetPath":"$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
                "optional":true
             },
             {
                "sourcePath":"$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
                "targetPath":"$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
                "optional":true
             },
             {
                "sourcePath":"$['urn:ietf:params:scim:schemas:extension:successfactors:2.0:User']['perPersonUuid']",
                "targetPath":"$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']['attributes'][0]['value']",
                "optional":true
             },
             {
                "targetPath":"$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']['attributes'][0]['name']",
                "condition":"$['urn:ietf:params:scim:schemas:extension:successfactors:2.0:User']['perPersonUuid'] EMPTY false",
                "constant":"customAttribute1"
             },
                "sourcePath":"$['urn:ietf:params:scim:schemas:extension:successfactors:2.0:User']['personIdExternal']",
                "targetPath":"$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']['attributes'][1]['value']",
                "optional":true
             },
             {
                "targetPath":"$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']['attributes'][1]['name']",
                "condition":"$['urn:ietf:params:scim:schemas:extension:successfactors:2.0:User']['personIdExternal'] EMPTY false",
                "constant":"customAttribute2"
             }
       ]
    },
    "group":{
          "mappings":[
     
     .....
    
  8. Run a second provisioning job, this time from SAP SuccessFactors, to replicate the identities to the same identity directory instance.

    This provisioning job will create new users in the Identity Directory from SAP SuccessFactors and update existing users with additional attributes from SAP SuccessFactors, specifically userType, department, division, perPersonUuid, and personIdExternal.

As a result, the identities from the two source systems are provisioned to a single target identity directory. Continuous provisioning jobs from both source systems run without overwriting attributes. Any updates are executed as patch requests.

Configure Local Identity Directory as a proxy system in the Identity Provisioning.

You can use the identity directory as a SCIM 2.0 based proxy connector. Before exposing it to an external system, you should first populate it with users and groups. To learn how to read and then provision them to the identity directory, see the Source-Target System Provisioning section above.

Note:

The Identity Provisioning implementation of the Proxy System SCIM API (based on the SCIM Query) supports single entity and delta read filtering for users and groups. For more information, see Query Parameters for Proxy System SCIM APIFilters.

  1. Sign in to SAP Cloud Identity Services administration console and create a technical user with the necessary authorizations. It will later be used by the external consumer to connect to Identity Provisioning.

    If you already have a technical user, skip this step.

    • For Certificate-based authentication, follow the procedure in Manage Certificates for Inbound ConnectionSAP Cloud Identity Infrastructure.

      For Basic authentication, navigate to Users & Authorizations > Administrators and add an admin user of type System. Configure a client ID and secret for this user and enable the Access Proxy System API permission.

  2. Navigate to Identity Provisioning > Proxy Systems.

  3. Add Local Identity Directory as a proxy system.

  4. Optional: Configure properties applicable for the Local Identity Directory proxy system, such as idds.user.filter and idds.group.filter. For more information, see List of Properties.

  5. Optional: Adapt the default proxy read and proxy write transformations:

    Read Transformation

    Write Transformation

    Code Syntax:

    {
      "user": {
        "scimEntityEndpoint": "Users",
        "mappings": [
          {
            "sourcePath": "$.id",
            "targetVariable": "entityIdSourceSystem",
            "targetPath": "$.id"
          },
          {
            "sourceVariable": "entityBaseLocation",
            "targetVariable": "entityLocationSourceSystem",
            "targetPath": "$.meta.location",
            "functions": [
              {
                "type": "concatString",
                "suffix": "${entityIdSourceSystem}"
              }
            ]
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['userUuid']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['userUuid']"
          },
          {
            "sourcePath": "$.schemas",
            "preserveArrayWithSingleElement": true,
            "targetPath": "$.schemas"
          },
          {
            "sourcePath": "$.userName",
            "targetPath": "$.userName",
            "optional": true,
            "correlationAttribute": true
          },
          {
            "sourcePath": "$.emails[*].value",
            "preserveArrayWithSingleElement": true,
            "targetPath": "$.emails[?(@.value)]"
          },
          {
            "sourcePath": "$.emails[0].value",
            "targetPath": "$.emails[0].value"
          },
          {
            "sourcePath": "$.emails[?(@.primary== true)].value",
            "correlationAttribute": true
          },
          {
            "sourcePath": "$.active",
            "targetPath": "$.active"
          },
          {
            "sourcePath": "$.userType",
            "targetPath": "$.userType",
            "optional": true
          },
          {
            "sourcePath": "$.name.givenName",
            "targetPath": "$.name.givenName",
            "optional": true
          },
          {
            "sourcePath": "$.name.middleName",
            "targetPath": "$.name.middleName",
            "optional": true
          },
          {
            "sourcePath": "$.name.familyName",
            "targetPath": "$.name.familyName",
            "optional": true
          },
          {
            "sourcePath": "$.name.honorificPrefix",
            "targetPath": "$.name.honorificPrefix",
            "optional": true
          },
          {
            "sourcePath": "$.addresses",
            "targetPath": "$.addresses",
            "preserveArrayWithSingleElement": true,
            "optional": true
          },
          {
            "sourcePath": "$.locale",
            "targetPath": "$.locale",
            "optional": true
          },
          {
            "sourcePath": "$.phoneNumbers",
            "targetPath": "$.phoneNumbers",
            "preserveArrayWithSingleElement": true,
            "optional": true
          },
          {
            "sourcePath": "$.timezone",
            "targetPath": "$.timezone",
            "optional": true
          },
          {
            "sourcePath": "$.displayName",
            "targetPath": "$.displayName",
            "optional": true
          },
          {
            "sourcePath": "$.sourceSystem",
            "targetPath": "$.sourceSystem",
            "ignore": true
          },
          {
            "sourcePath": "$.groups",
            "targetPath": "$.groups",
            "preserveArrayWithSingleElement": true,
            "optional": true
          },
          {
            "type": "remove",
            "targetPath": "$.groups[*].display"
          },
          {
            "condition": "$.displayName EMPTY true",
            "type": "remove",
            "targetPath": "$.displayName"
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']",
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:User']",
            "optional": true
          },
          {
            "sourcePath": "$.company",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
            "optional": true
          }
        ]
      },
      "group": {
        "scimEntityEndpoint": "Groups",
        "mappings": [
          {
            "sourcePath": "$.id",
            "targetPath": "$.id",
            "targetVariable": "entityIdSourceSystem"
          },
          {
            "sourceVariable": "entityBaseLocation",
            "targetVariable": "entityLocationSourceSystem",
            "targetPath": "$.meta.location",
            "functions": [
              {
                "type": "concatString",
                "suffix": "${entityIdSourceSystem}"
              }
            ]
          },
          {
             "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
             "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']"
          },
          {
             "sourcePath": "$.displayName",
             "targetPath": "$.displayName"
          },
          {
             "sourcePath": "$.members",
             "targetPath": "$.members",
             "preserveArrayWithSingleElement": true,
             "optional": true
          },
          {
            "sourcePath": "$.schemas",
            "preserveArrayWithSingleElement": true,
            "targetPath": "$.schemas"
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']",
            "optional": true
          }
        ]
      }
    }
    

    Code Syntax:

    {
      "user": {
        "scimEntityEndpoint": "Users",
        "mappings": [
          {
            "sourceVariable": "entityIdTargetSystem",
            "targetPath": "$.id"
          },
          {
            "constant": ["urn:ietf:params:scim:schemas:core:2.0:User","urn:ietf:params:scim:schemas:extension:enterprise:2.0:User","urn:ietf:params:scim:schemas:extension:sap:2.0:User"],
            "targetPath": "$.schemas"
          },
          {
            "sourcePath": "$.userName",
            "targetPath": "$.userName",
            "optional": true
          },
          {
            "sourcePath": "$.emails[*].value",
            "preserveArrayWithSingleElement": true,
            "targetPath": "$.emails[?(@.value)]"
          },
          {
            "sourcePath": "$.userType",
            "targetPath": "$.userType",
            "defaultValue": "employee",
            "optional": true
          },
          {
            "sourcePath": "$.name.givenName",
            "targetPath": "$.name.givenName",
            "optional": true
          },
          {
            "sourcePath": "$.name.middleName",
            "targetPath": "$.name.middleName",
            "optional": true
          },
          {
            "sourcePath": "$.name.familyName",
            "targetPath": "$.name.familyName",
            "optional": true
          },
          {
            "sourcePath": "$.name.honorificPrefix",
            "targetPath": "$.name.honorificPrefix",
            "optional": true
          },
          {
            "sourcePath": "$.addresses",
            "targetPath": "$.addresses",
            "preserveArrayWithSingleElement": true,
            "defaultValue": [],
            "optional": true,
            "functions": [
              {
                "function": "putIfAbsent",
                "key": "type",
                "defaultValue": "work"
              },
              {
                "condition": "(@.type NIN ['work', 'home'])",
                "function": "putIfPresent",
                "key": "type",
                "defaultValue": "work"
              }
            ]
          },
          {
            "sourcePath": "$.locale",
            "targetPath": "$.locale",
            "optional": true
          },
          {
            "sourcePath": "$.phoneNumbers",
            "targetPath": "$.phoneNumbers",
            "preserveArrayWithSingleElement": true,
            "optional": true
          },
          {
            "sourcePath": "$.displayName",
            "targetPath": "$.displayName",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validFrom']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validFrom']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validTo']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['validTo']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['employeeNumber']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['costCenter']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['organization']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['division']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['department']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['value']",
            "optional": true
          },
          {
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:enterprise:2.0:User']['manager']['displayName']",
            "optional": true
          },
          {
            "sourcePath": "$.active",
            "targetPath": "$.active",
            "defaultValue": true,
            "optional": true
          },
          {
            "constant": false,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sendMail']",
            "scope": "createEntity"
          },
          {
            "sourcePath": "$.emails",
            "preserveArrayWithSingleElement": true,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails']",
            "scope": "createEntity",
            "functions": [
              {
                "function": "putIfAbsent",
                "key": "verified",
                "defaultValue": true
              }
            ]
          },
          {
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['emails'][*]['type']",
            "type": "remove"
          },
          {
            "constant": "disabled",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['passwordDetails']['status']",
            "scope": "createEntity"
          },
          {
            "constant": 39,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:User']['sourceSystem']",
            "scope": "createEntity"
          },
          {
            "sourcePath": "$.timezone",
            "targetPath": "$.timezone",
            "optional": true
          },
          {
            "sourcePath": "$.Operations",
            "targetPath": "$.Operations",
            "preserveArrayWithSingleElement": true,
            "scope": "patchEntity"
          },
          {
            "sourcePath": "$.schemas",
            "targetPath": "$.schemas",
            "preserveArrayWithSingleElement": true,
            "scope": "patchEntity"
          }
        ]
      },
      "group": {
        "scimEntityEndpoint": "Groups",
        "mappings": [
          {
            "sourceVariable": "entityIdTargetSystem",
            "targetPath": "$.id"
          },
          {
            "sourcePath": "$.Operations",
            "targetPath": "$.Operations",
            "preserveArrayWithSingleElement": true,
            "scope": "patchEntity"
          },
          {
            "sourcePath": "$.schemas",
            "targetPath": "$.schemas",
            "preserveArrayWithSingleElement": true,
            "scope": "patchEntity"
          },
          {
            "constant": ["urn:ietf:params:scim:schemas:core:2.0:Group","urn:sap:cloud:scim:schemas:extension:custom:2.0:Group","urn:ietf:params:scim:schemas:extension:sap:2.0:Group"],
            "targetPath": "$.schemas"
          },
          {
            "sourcePath": "$.displayName",
            "targetPath": "$.displayName"
          },
          {
            "sourcePath": "$.members",
            "targetPath": "$.members",
            "preserveArrayWithSingleElement": true,
            "optional": true
          },
          {
            "sourcePath": "$.displayName",
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
            "scope": "createEntity",
            "functions": [
              {
                "type": "replaceAllString",
                "regex": "[\\s\\p{Punct}]",
                "replacement": "_"
              }
            ]
          },
          {
            "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
            "optional": true,
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['name']",
            "scope": "createEntity"
          },
          {
            "sourcePath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['description']",
            "optional": true,
            "targetPath": "$['urn:sap:cloud:scim:schemas:extension:custom:2.0:Group']['description']"
          },
          {
            "condition": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId'] EMPTY false",
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId']",
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId']"
          },
          {
            "condition": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId'] EMPTY false",
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']",
            "optional": true,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['type']"
          },
          {
            "condition": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['applicationId'] EMPTY false",
            "sourcePath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']",
            "optional": true,
            "targetPath": "$['urn:ietf:params:scim:schemas:extension:sap:2.0:Group']['supportedOperations']"
          }
        ]
      }
    }
    
  6. Connect the external consumer to Identity Provisioning with the technical user you have created in Step 1.

    If the external consumer system is SAP Identity Management, you can export the newly created proxy system as a SCIM repository from Identity Provisioning and import it in SAP Identity Management. This will create a SCIM repository in SAP Identity Management where most of the repository constants will be automatically filled in. You need to provide the technical user credentials that you have set up in Step 1 and the SCIM assignment method as described below:

    • For AUTH_USER and AUTH_PASSWORD, enter the user ID and password of the technical user for which you have set permission Access Proxy System API.

    • For the SCIM_ASSIGNMENT_METHOD constant, make sure the value is PUT.

    Note:

    For external consumer systems, other than SAP Identity Management, you should also use the PUT method for modifying entities.

Results

When a proxy system is connected to an external backend system (in the case of SAP Identity Management this means the exported CSV file is imported into the Identity Management Admin UI and a repository is configured), you can start managing the users and groups into this external system. Usually, the first operation is the initial load of the existing entities into your external system. When this load has finished, changes in the external system, such as creating new users or updating existing ones, can trigger CRUD requests back to the proxy system.