Authoring the SAP ECC 7.51 Web Service connector template for the ECMA2Host
This guide walks you through the process of creating a template for the Web Service Extensibility Connectivity Management Agent (ECMA) connector to manage SAP ECC users.
Limitations and assumptions
This template demonstrates how to manage users. Other object types like Local Activity Groups, Roles, and Profiles aren't covered by this guide as the ECMA2Host doesn't currently support multi-valued references. Password operations are also out of scope for this guide.
This guide doesn't cover the creation of the service account within SAP that is used to call the exposed BAPI functions. It assumes that a pre-created demo account Developer is used with a profile RFC_ALL that grants permissions to the BAPIs mentioned in this article.
The Web Service Configuration Tool doesn't support the following features exposed in SAP by default: WSP Policies and multiple bindings per endpoint. It does expect a WSDL with SOAP 1.1 only, all-in-one document style binding without policies.
SAP ECC BAPI functions used in this template:
- BAPI_USER_GETLIST - get a list of all users connected to this system.
- BAPI_USER_GETDETAIL - get details of specific user.
- BAPI_USER_CREATE1 - creates a user.
- BAPI_USER_DELETE - deletes a user.
- BAPI_USER_CHANGE - updates a user.
All SAP user properties in this guide are treated as single valued properties.
The programming language used is Visual Basic.
Defining a web service endpoint and creating a schema
Before you can design import and export workflows, you need to create a template and define an endpoint with the SAP BAPI functions exposed over a SOAP interface. Then create a schema of the ECMA2 objects and their properties is available in this template.
- From the "C:\Program Files\Microsoft ECMA2Host\Web Service Configuration Tool" folder, start the Web Service Configuration tool wsconfigTool.exe
- From the File-New menu, choose Create new SOAP Project
- Select on SOAP Project and choose Add new Web Service.
- Name your web service SAPECC, provide a URL to download WSDL published, enter SAPECC as namespace. Web Service name helps you to distinguish this web service in your template from others. Namespace defines a name of the Microsoft .NET namespace used to generate classes. Choose Basic authentication mode unless instructed otherwise by SAP administrator. Select Next.
- Provide credentials to connect to SAP ECC endpoint. Select Next.
- On the endpoints and operations page, ensure that the BAPIs are displayed and select Finish
Note
if you see more than one endpoint, then you have both SOAP 1.2 and SOAP 1.1 bindings enabled. This causes the connector to fail. Modify your binding definition in SOAMANAGER and keep only one. Then re-add a web service.
- Save the project into C:\Program Files\Microsoft ECMA2Host\Service\ECMA folder.
- Select on Object Types tab and choose to add User object type. Select Ok.
- Expand Object Types tab and select on User type definition.
- Add the following attributes into schema and choose userName as the anchor.
- Save your project.
Name | Type | Anchor |
---|---|---|
city | string | |
company | string | |
department | string | |
string | ||
expirationTime | string | |
firstName | string | |
lastName | string | |
middleName | string | |
telephoneNumber | string | |
jobTitle | string | |
userName | string | checked |
Creating Full Import workflow
The Import workflow, while being optional in the ECMA2Host, allows you to import existing SAP users into the ECMA2Host in-memory cache and avoid duplicate users being created during provisioning.
If you don't create an import workflow, then your connector is operating in Export-only mode and causes the ECMA2Host to always issue Create user operations, even for existing users. This may lead to failures or duplicates when standard SAP BAPIs are used unless duplicates are handled by the export workflow.
SAP ECC doesn't offer a built-in mechanism for reading changes made since the last read.
Therefore, we're implementing the Full Import workflow only. Should you need to implement Delta Imports for performance reasons, consult your SAP administrator for a list of BAPIs and have them published as a SOAP webservice. Then, implement the Delta Import workflow using the following approach as described and a customData property that contains a timestamp of the previous successful run.
SAP ECC offers several BAPI functions to get a list of users with their properties:
- BAPI_USER_GETLIST - get a list of all users connected to this system.
- BAPI_USER_GETDETAIL - get details of specific user.
Only these two BAPIs are used to retrieve existing users from SAP ECC in this template.
- Navigate to Object Types -> User -> Import -> Full Import workflow and from the Toolbox on the right drag and drop Sequence activity onto workflow designer pane.
- On the bottom left, find the Variables button and select it to expand a list of variables defined within this Sequence.
- Add the following variables. To select a variable type generated from the SAP WSDL, select to Browse for Types and expand generated and then expand SAPECC namespace.
Name | Variable Type | Scope | Default |
---|---|---|---|
selRangeTable | SAPECC.TABLE_OF_BAPIUSSRGE | Sequence | new TABLE_OF_BAPIUSSRGE with {.item = new BAPIUSSRGE(){new BAPIUSSRGE}} |
getListRetTable | SAPECC.TABLE_OF_BAPIRET2 | Sequence | new TABLE_OF_BAPIRET2 |
pageSize | Int32 | Sequence | 200 |
returnedSize | Int32 | Sequence | |
usersTable | SAPECC.TABLE_OF_BAPIUSNAME | Sequence | new TABLE_OF_BAPIUSNAME() |
- From the Toolbox, drag and drop four Assign activities inside your Sequence activity and set these values:
selRangeTable.item(0).PARAMETER = "USERNAME"
selRangeTable.item(0).SIGN = "I" selRangeTable.item(0).OPTION = "GT" selRangeTable.item(0).LOW = ""
These parameters used to call the BAPI_USER_GETLIST function and to implement pagination.
- To implement pagination, from the Toolbox drag and drop the DoWhile activity inside your Sequence activity after the last Assign operation.
- On the right pane switch to the Properties tab and enter this condition for the DoWhile
- cycle:
returnedSize = pageSize
- Select on the Variables and add currentPageNumber property of int32 type within DoWhile cycle with default value of 0.
- Optional step: if you plan to implement Delta Import workflow, then from the Toolbox drag and drop Assign activity inside your Sequence activity after DoWhile cycle. Set this value:
customData(schemaType.Name + "_lastImportTime") = DateTimeOffset.UtcNow.Ticks.ToString()
This saves the date and time of the last full import run and this timestamp can later be used in Delta Import workflow.
- From the Toolbox, drag and drop Sequence activity inside your DoWhile activity. Drag and drop WebServiceCall activity inside that Sequence activity and select SAPECC service name, ZSAPCONNECTORWS endpoint and BAPI_USER_GETLIST operation.
- Select on ... Arguments button to define parameters for web service call as follows:
Name | Direction | Type | Value |
---|---|---|---|
MAX_ROWS | In | Int32 | pageSize |
MAX_ROWSSpecified | In | Boolean | True |
RETURN | In/Out | TABLE_OF_BAPIRET2 | getListRetTable |
SELECTION_EXP | In/Out | TABLE_OF_BAPIUSSEXP | |
SELECTION_RANGE | In/Out | TABLE_OF_BAPIUSSRGE | selRangeTable |
USERLIST | In/Out | TABLE_OF_BAPIUSNAME | usersTable |
WITH_USERNAME | In | String | |
ROWS | Out | Int32 | returnedSize |
- Select OK. The warning sign disappears. The list of users stored in the usersTable variable. As SAP doesn't return a complete list of users in one single response, we need to implement pagination and call this function several times while switching pages. Then for every user imported you need to get that user's details by making a separate call. That means that for a landscape with 1,000 users and a page size of 200, Web Service connector makes 5 calls to retrieve a list of users and 1,000 individual calls to retrieve users' details. To improve performance, ask your SAP team to develop a custom BAPI program that lists all uses with their properties. This avoids the need of making 1,000 individual calls and expose that BAPI function over SOAP WS endpoint.
- From the Toolbox, drag and drop IF activity inside your DoWhile activity after WebServiceCall activity. Specify this condition to check for non-empty response and absence of errors:
IsNothing(getListRetTable.item) OrElse getListRetTable.item.Count(Function(errItem) errItem.TYPE.Equals("E") = True) = 0
- From the Toolbox, drag and drop Throw activity into Else branch of your IF activity to throw an error on unsuccessful import. Switch to the Properties tab and enter this expression for Exception property of the Throw activity:
New Exception(getListRetTable.item.First(Function(retItem) retItem.TYPE.Equals("E")).MESSAGE)
- To process a list of imported users, drag and drop ForEachWithBodyFactory activity from the Toolbox into Then branch of your IF activity. Switch to Properties tab and select SAPECC.BAPIUSNAME as TypeArgument. Select on ... button and type this expression for values property:
if(usersTable.item,Enumerable.Empty(of BAPIUSNAME)())
- From the Toolbox, drag and drop Sequence activity inside your ForEach activity. Having this Sequence activity window active, select on the Variables button and define these variables:
Name | Variable Type | Scope | Default |
---|---|---|---|
company | SAPECC.BAPIUSCOMP | Sequence | new BAPIUSCOMP() |
address | SAPECC.BAPIADDR3 | Sequence | new BAPIADDR3() |
defaults | SAPECC.BAPIDEFAUL | Sequence | new BAPIDEFAUL() |
logondata | SAPECC.BAPILOGOND | Sequence | new BAPILOGOND() |
getDetailRetTable | SAPECC.TABLE_OF_BAPIRET2 | Sequence | new TABLE_OF_BAPIRET2() |
Your IF activity looks like this:
- Drag and drop CreateCSEntryChangeScope activity inside your Sequence activity. In the DN property, enter schemaType.Name & item.USERNAME. In the CreateAnchorAttribute AnchorValue field enter item.username.
- To retrieve details of each user, from the Toolbox drag and drop WebServiceCall activity inside Sequence activity right before CreateAnchorAttribute activity. Select SAPECC service name, ZSAPCONNECTORWS endpoint and BAPI_USER_GET_DETAIL operation. Select on ... Arguments button to define parameters for web service call as follows:
Name | Direction | Type | Value |
---|---|---|---|
RETURN | In/Out | TABLE_OF_BAPIRET2 | getDetailRetTable |
USERNAME | In | String | item.username |
ADDRESS | Out | BAPIADDR3 | address |
COMPANY | Out | BAPIUSCOMP | company |
DEFAULTS | Out | BAPIUSDEFAUL | defaults |
LOGONDATA | Out | BAPILOGOND | logonData |
WITH_USERNAME | In | String | |
ROWS | Out | Int32 | returnedSize |
- Select OK. The warning sign disappears. The details of a user are stored in the above listed variables. Your IF activity looks like this:
- To check the results of the BAPI_USER_GET_DETAIL operation, from the Toolbox, drag and drop IF activity and place it inside the Sequence activity in between WebServiceCall and CreateAnchorAttribute activities. Enter this condition:
IsNothing(getDetailRetTable.item) OrElse getDetailRetTable.item.Count(Function(errItem) errItem.TYPE.Equals("E") = True) = 0
As missing user details shouldn't be treated as a catastrophic event, we want to indicate this error and continue processing of other users. Drag and drop Sequence activity into Else branch of your IF activity. Add Log activity inside that new Sequence activity. Switch to Properties tab and change Level property to High, Tag to Trace. Enter the following into LogText property: string.Join("\n", getDetailRetTable.item.Select (Function(item) item.MESSAGE ))
- Drag and drop Sequence activity into Then branch of IF activity. Drag and drop existing CreateAnchorAttribute activity to Sequence activity inside Then branch of IF activity. Your ForEach activity now looks like this:
- For each property of a user like city, company, department, email add IF activity after CreateAnchorAttribute activity and check for non-empty values by entering conditions like
Not string.IsNullOrEmpty(address.city)
and adding CreateAttributeChange activities into Then branch of that IF activity.
For example: Add CreateAttributeChange activities for all user properties using this mapping table:
ECMA User property | SAP property |
---|---|
city | address.city |
department | address.department |
company | company.company |
address.e_mail | |
firstName | address.firstName |
lastName | address.lastName |
middleName | address.middleName |
jobTitle | address.function |
expirationTime | logonData.GLTGB |
telephoneNumber | address.TEL1_NUMBR |
- Finally, add SetImportStatusCode activity after the last CreateAttributeChange activity. Set ErrorCode to Success in the Then branch. Add one more SetImportStatus code activity into Else branch and set ErrorCode to ImportErrorCustomContinueRun.
- Collapse Sequence activity inside ForEach activity so your DoWhile cycle looks like this:
- To retrieve the next page of users, update
selRangeTable.item(0).LOW
property. Drag and drop IF activity into Sequence activity within DoWhile. Place it after existing IF activity. Enter returnedSize>0 as Condition. Add Assign activity into Then branch of IF activity and setselRangeTable.item(0).LOW
tousersTable.item(returnedSize-1).username
.
You completed the definition of Full Import workflow.
Creating Export Add workflow
To create a user in SAP ECC you can call BAPI_USER_CREATE1 program and provide all the parameters including an account name and an initial password. If you need an account name to be generated on the SAP side, consult with your SAP administrator and use a custom BAPI function that returns a userName property of a newly created user account.
This guide doesn't demonstrate assignment of licenses, local or global activity groups, systems or profiles. Consult with your SAP administrator and modify this workflow accordingly.
There's no need to implement pagination in export workflows. There's only one object objectToExport available within the workflow context.
- Navigate to Object Types -> User -> Export -> Add workflow and from the Toolbox on the right drag and drop Sequence activity onto workflow designer pane.
- On the bottom left, find the Variables button and select it to expand a list of variables defined within this Sequence.
- Add the following variables. To select a variable type generated from the SAP WSDL, select to Browse for Types and expand generated and then expand SAPECC namespace. This initializes the data structures used by BAPI_USER_CREATE1 program.
Name | Variable Type | Scope | Default |
---|---|---|---|
address | SAPECC.BAPIADDR3 | Sequence | new BAPIADDR3() |
userName | String | Sequence | |
password | SAPECC.BAPIPWD | Sequence | new BAPIPWD() |
company | SAPECC.BAPIUSCOMP | Sequence | new BAPIUSCOMP() |
defaults | SAPECC.BAPIDEFAUL | Sequence | new BAPIDEFAUL() |
logOnData | SAPECC.BAPILOGOND | Sequence | new BAPILOGOND() |
bapiret2Table | SAPECC.TABLE_OF_BAPIRET2 | Sequence | new TABLE_OF_BAPIRET2() |
- As we defined userName property as an immutable ID, an anchor, we need to extract userName value from a collection of anchors of our export object. Drag and drop ForEachWithBodyFactory activity from the Toolbox into your Sequence activity. Replace item variable name with anchor, switch to properties and choose TypeArgument of
Microsoft.MetadirectoryServices.AnchorAttribute
. In the Value field, typeobjectToExport.AnchorAttributes
.
- To extract a string value of a userName anchor, drag and drop Switch activity inside the ForEach activity. In the popup window, select
Microsoft.IdentityManagement.MA.WebServices.Activities.Extensions.AnchorAttributeNameWrapper
type of a switch. Enter Expression value of: New AnchorAttributeNameWrapper(anchor.Name). - Select on Add new case area of Switch activity. Type userName as Case Value. Drag and drop Assign activity into userName case body and assign anchor.Value.ToString() to userName variable.
- Now that we extracted userName value from exported object anchor property, we need to populate other structures like company, defaults, address, logon data that contain other SAP user details. We do this by cycling through collection of attribute changes.
- Collapse your ForEach activity and drag and drop another ForEachWithBothFactory activity inside your Sequence activity after the existing ForEach activity. Replace item variable name with attributeChange, switch to properties and choose TypeArgument of
Microsoft.MetadirectoryServices.AttributeChange
. In the Value field, typeobjectToExport.AttributeChanges
.
- Drag and drop the Switch activity into the body of your ForEach activity.
- In the popup menu, select
Microsoft.IdentityManagement.MA.WebServices.Activities.Extensions.AttributeNameWrapper
and select Ok. - Enter the following expression: New AttributeNameWrapper(attributeChange.Name). You'll see a warning icon in the top right corner of your Switch activity about unhandled attributes defined in schema and not assigned to any property.
- Select on Add new case area of Switch activity and type a case value of city.
- Drag and drop Assign activity into the body of this case. Assign
attributeChange.ValueChanges(0).Value.ToString()
to address.city.
- Add other missing cases and assignments. Use this mapping table as a guide:
Case | Assignment |
---|---|
city | address.city = attributeChange.ValueChanges(0)Value.ToString() |
company | company.company = attributeChange.ValueChanges(0)Value.ToString() |
department | address.department = attributeChange.ValueChanges(0)Value.ToString() |
address.e_mail = attributeChange.ValueChanges(0)Value.ToString() | |
expirationTime | logOnData.GLTGB = attributeChange.ValueChanges(0)Value.ToString() |
firstname | address.firstname = attributeChange.ValueChanges(0)Value.ToString() |
lastName | address.lastname = attributeChange.ValueChanges(0)Value.ToString() |
middleName | address.middlename = attributeChange.ValueChanges(0)Value.ToString() |
telephoneNumber | address.TEL1_Numbr = attributeChange.ValueChanges(0)Value.ToString() |
jobTitle | address.function = attributeChange.ValueChanges(0)Value.ToString() |
export_password | password.BAPIPWD1 = attributeChange.ValueChanges(0)Value.ToString() |
Here, export_password is a special virtual attribute that is always defined in the schema and can be used to pass an initial password of user being created.
- Collapse your ForEach activity and drag and drop the IF activity into the Sequence activity, after the second ForEach activity, to validate the user properties, before submitting the create user request. We need at least 3 non-empty values: username, last name, initial password. Enter this condition:
(String.IsNullOrEmpty(address.lastname) = False ) AND (String.IsNullOrEmpty(userName) = False) AND (String.IsNullOrEmpty(password.BAPIPWD1) = False)
- In the Else branch of IF activity, add one more IF activity as we want to throw different errors depending on what is missing. Enter condition value: String.IsNullOrEmpty(userName). Drag and Drop
CreateCSEntryChangeResult
activities into both branches of the second IF activity and set up ErrorCode ofExportErrorMissingAnchorComponent
andExportErrorMissingProvisioningAttribute
.
- Drag and drop Sequence activity in the empty Then branch of the first IF activity. Drag and drop WebSeviceCall activity inside the Sequence activity. Select SAPECC service name, ZSAPCONNECTORWS endpoint and BAPI_USER_CREATE1 operation. Select on ... Arguments button to define parameters for web service call as follows:
Name | Direction | Type | Value |
---|---|---|---|
ADDRESS | In | BAPIADDR3 | address |
COMPANY | In | BAPIUSCOMP | company |
DEFAULTS | In | BAPIDEFAUL | defaults |
LOGONDATA | In | BAPILOGOND | logOnData |
PASSWORD | In | BAPIPWD | password |
RETURN | In-Out | TABLE_OF_BAPIRET2 | bapiret2Table |
SELF_REGISTER | In | String | "X" |
USERNAME | In | String | userName |
- Select OK. The warning sign disappears.
- To process create user request results, drag and drop IF activity inside the Sequence activity after the WebServiceCall activity. Enter the following condition:
IsNothing (bapiret2Table.item) OrElse bapiret2Table.item.Count(Function(errItem) errItem.TYPE.Equals("E") = True) <> 0
- If we get no errors, we assume that export operation completed successfully, and we want to indicate successful export of this object by creating CSEntryChangeResult with Success status. Drag and drop CreateCSEntryChangeResult activity into Else branch of your IF activity and select Success error code.
- Optional: If the web service call returns a generated account name of a user, we need to update an anchor value of exported object. To do that, drag and drop
CreateAttrubuteChange
activity inside theCreateCSEntryChangeResult
activity and select to Add a userName. Then drag and dropCreateValueChange
activity inside theCreateAttributeChange
activity and enter the variable name populated by a web service call activity. In this guide, you use the userName variable that isn't updated on export.
- The last step in the Export Add workflow is to handle and log export errors. Drag and drop Sequence activity into the empty Then branch of your IF activity.
- Drag and drop Log activity into Sequence activity. Switch to Properties tab and enter LogText value of:
bapiret2Table.item.First(Function(retItem) retItem.TYPE.Equals("E"))
.MESSAGE. Keep High logging level and Trace tag. This logs an error message into ConnectorsLog or the ECMA2Host event log when verbose tracing is enabled. - Drag and drop Switch activity inside the Sequence activity after Log activity. In the popup window, select String type of the switch value. Enter the following expression:
bapiret2Table.item.First(Function(retItem) retItem.TYPE.Equals("E")).NUMBER
- Select on Default case and drag and drop CreateCSEntryChangeResult activity into the body of this case. Choose ExportErrorInvalidProvisioningAttributeValue error code.
- Select on the Add new case area and type a case value of 224. Drag and drop
CreateCSEntryChangeResult
activity into the body of this case. ChooseExportErrorCustomContinueRun
error code.
You completed the definition of Export Add workflow.
Creating Export Delete workflow
To delete a user in SAP ECC, you can call BAPI_USER_DELETE program and provide an account name to be deleted in connected system. Consult with your SAP Administrator to determine whether this scenario is mandatory. Most often, SAP ECC accounts aren't deleted but are set to expire to keep historical records.
This guide doesn't cover scenarios related to SAP Common User Administration system, deprovisioning of users from connected systems, revocation of licenses, etc.
There's no need to implement pagination in export workflows. There's only one object objectToExport available within the workflow context.
- Navigate to Object Types -> User -> Export -> Delete workflow and from the Toolbox on the right drag and drop Sequence activity onto workflow designer pane.
- On the bottom left, find the Variables button and select it to expand a list of variables defined within this Sequence.
- Add the following variables. To select a variable type generated from the SAP WSDL, select to Browse for Types and expand generated and then expand SAPECC namespace. This initializes the data structures used by BAPI_USER_DELETE program.
Name | Variable Type | Scope | Default |
---|---|---|---|
userName | String | Sequence | |
bapiret2Table | SAPECC.TABLE_OF_BAPIRET2 | Sequence | new TABLE_OF_BAPIRET2() |
- As we defined userName property as an immutable ID, an anchor, we need to extract userName value from a collection of anchors of our export object. Drag and drop ForEachWithBodyFactory activity from the Toolbox into your Sequence activity. Replace item variable name with anchor, switch to properties and choose TypeArgument of
Microsoft.MetadirectoryServices.AnchorAttribute
. In the Value field, typeobjectToExport.AnchorAttributes
.
- To extract a string value of a userName anchor, drag and drop Switch activity inside the ForEach activity. In the popup window, select
Microsoft.IdentityManagement.MA.WebServices.Activities.Extensions.AnchorAttributeNameWrapper
type of a switch. Enter Expression value of: NewAnchorAttributeNameWrapper(anchor.Name)
. Select on Add new case area of Switch activity. Type userName as Case Value. Drag and drop Assign activity into userName case body and assignanchor.Value.ToString()
to userName variable. - Drag and drop WebSeviceCall activity inside the Sequence activity after ForEach activity. Select SAPECC service name, ZSAPCONNECTORWS endpoint and BAPI_USER_DELETE operation. Select on ... Arguments button to define parameters for web service call as follows:
Name | Direction | Type | Value |
---|---|---|---|
RETURN | In/Out | TABLE_OF_BAPIRET2 | bapiret2Table |
USERNAME | In | String | userName |
- Select OK. The warning sign disappears.
- To process delete user request results drag and drop IF activity inside the Sequence activity after the WebServiceCall activity. Enter the following condition:
If(bapiRet2Table.item, Enumerable.Empty(Of BAPIRET2)()).Count(Function(errItem) errItem.TYPE.Equals("E") = True) <> 0
- If we get no errors, we assume that delete operation completed successfully and we want to indicate successful export of this object by creating
CSEntryChangeResult
with Success status. Drag and dropCreateCSEntryChangeResult
activity into Else branch of your IF activity and select Success error code.
- The last step in the Export Delete workflow is to handle and log export errors. Drag and drop Sequence activity into the empty Then branch of your IF activity.
- Drag and drop Log activity into Sequence activity. Switch to Properties tab and enter LogText value of:
bapiRetTable.item.First(Function(retItem) retItem.TYPE.Equals("E")= True).MESSAGE
. Keep High logging level and Trace tag. This logs an error message into the ConnectorsLog or ECMA2Host event log when verbose tracing is enabled. - Drag and drop Switch activity inside the Sequence activity after Log activity. In the popup window, select String type of the switch value. Enter the following expression:
bapiret2Table.item.First(Function(retItem) retItem.TYPE.Equals("E")).NUMBER
- Select on Default case and drag and drop CreateCSEntryChangeResult activity into the body of this case. Choose ExportErrorSyntaxViolation error code.
- Select on the Add new case area and type a case value of 124. Drag and drop
CreateCSEntryChangeResult
activity into the body of this case. ChooseExportErrorCustomContinueRun
error code.
You completed the definition of Export Delete workflow.
Creating Export Replace workflow
To update a user in SAP ECC, you can call BAPI_USER_CHANGE program and provide all the parameters including an account name and all user details, including those that aren't changing. The ECMA2 export mode when all user properties are to be provided is called Replace. In comparison, export mode of AttributeUpdate only provides you with attributes that are being changed and this may cause some user properties to be overwritten with empty values. Therefore, Webservice connector always uses Object Replace export mode and expects the connector to be configured for Export Type: Replace.
Export Replace workflow is almost identical to the Export Add workflow. The only difference is that you need to specify extra parameters like addressX or companyX for the BAPI_USER_CHANGE program. The X at the end of addressX indicates that the structure of the address does contain a change.
- Navigate to Object Types -> User -> Export -> Replace workflow and from the Toolbox on the right drag and drop Sequence activity onto workflow designer pane.
- On the bottom left, find the Variables button and select it to expand a list of variables defined within this Sequence.
- Add the following variables. To select a variable type generated from the SAP WSDL, select to Browse for Types and expand generated and then expand SAPECC namespace. This initializes the data structures used by BAPI_USER_CHANGE program.
Name | Variable Type | Scope | Default |
---|---|---|---|
userName | String | Sequence | |
bapiret2Table | SAPECC.TABLE_OF_BAPIRET2 | Sequence | new TABLE_OF_BAPIRET2() |
addressX | SAPECC.BAPIADDR3X | Sequence | new BAPIADDR3X() |
address | SAPECC.BAPIADDR3 | Sequence | new BAPIADDR3() |
companyX | SAPECC. BAPIUSCOMX | Sequence | new BAPIUSCOMX() |
company | SAPECC.BAPIUSCOMP | Sequence | new BAPIUSCOMP() |
defaultsX | SAPECC.BAPIDEFAX | Sequence | new BAPIDEFAX() |
defaults | SAPECC.BAPIDEFAUL | Sequence | new BAPIDEFAUL() |
logOnDataX | SAPECC.BAPILOGONX | Sequence | new BAPILOGONX() |
logOnData | SAPECC.BAPILOGOND | Sequence | new BAPILOGOND() |
Your Export Replace workflow looks like this:
- As we defined userName property as an immutable ID, an anchor, we need to extract userName value from a collection of anchors of our export object. Drag and drop ForEachWithBodyFactory activity from the Toolbox into your Sequence activity. Replace item variable name with anchor, switch to properties and choose TypeArgument of
Microsoft.MetadirectoryServices.AnchorAttribute
. In the Value field, typeobjectToExport.AnchorAttributes
.
- To extract a string value of a userName anchor, drag and drop Switch activity inside the ForEach activity. In the popup window, select
Microsoft.IdentityManagement.MA.WebServices.Activities.Extensions.AnchorAttributeNameWrapper
type of a switch. Enter Expression value of: NewAnchorAttributeNameWrapper(anchor.Name)
. Select on Add new case area of Switch activity. Type userName as Case Value. Drag and drop Assign activity into userName case body and assignanchor.Value.ToString()
to userName variable. Your Export Replace workflow looks like this:
- Now that we extracted userName value from exported object anchor property, we need to populate other structures like company, defaults, address, logon data that contain other SAP user details. We do this by cycling through collection of all attributes defined in the schema.
- Collapse your ForEach activity and drag and drop another ForEachWithBothFactory activity inside your Sequence activity after the existing ForEach activity. Replace item variable
name with schemaAttr, switch to properties and choose TypeArgument of
Microsoft.MetadirectoryServices.SchemaAttribute
. In the Value field, typeschemaType.Attributes
.
- Drag and drop the Sequence activity into the body of your ForEach activity. On the bottom left, find the Variables button and select it to expand a list of variables defined within this Sequence. Add the following variable: xValue of String type. Drag and drop Assign activity into your Sequence activity. Assign xValue the expression of:
If(objectToExport.AttributeChanges.Contains(schemaAttr.Name), objectToExport.AttributeChanges(schemaAttr.Name).ValueChanges(0).Value.ToString(), String.Empty)
It either extracts the changes staged for export for this attribute, or initializes it with an empty string. Your Export Replace workflow looks like this:
- Drag and drop the Switch activity after Assign activity. In the popup menu, select
Microsoft.IdentityManagement.MA.WebServices.Activities.Extensions.AttributeNameWrapper
and select Ok. Enter the following expression: NewAttributeNameWrapper(schemaAttr.Name)
. You'll see a warning icon in the top right corner of your Switch activity about unhandled attributes defined in the schema and not assigned to any property. Select on Add new case area of Switch activity and type a case value of city. Drag and drop Sequence activity into the body of this case. Drag and drop the Assign activity into the body of this case. Assign "X" value to addressX.city. Drag and drop another Assign activity into the body of this case. Assign xValue to address.city. Your Export Replace workflow looks like this:
10.Add other missing cases and assignments. Use this mapping table as a guide:
Case | Assignment |
---|---|
city | addressX.city = "X" address.city = xValue |
company | companyX.company = "X" company.company = xValue |
department | address.departmentX = "X" address.department = xValue |
addressX.e_mail = "X" address.e_mail = xValue | |
expirationTime | logOnDataX.GLTGB = "X" logOnData.GLTGB = xValue |
firstname | addressX.firstname = "X" address.firstname = xValue |
lastName | addressX.lastname = "X" address.lastname = xValue |
middleName | addressX.middlename = "X" address.middlename = xValue |
telephoneNumber | addressX.TEL1_Numbr = "X" address.TEL1_Numbr = xValue |
jobTitle | addressX.function = "X" address.function = xValue |
Your Export Replace workflow looks like this:
Before calling BAPI_USER_CHANGE program, we need to check for non-empty username. Collapse both ForEach activities and drag and drop IF activity after the second ForEach activity. Enter the following condition:
String.IsNullOrEmpty(userName ) = False
When username is empty, we want to indicate that operation was unsuccessful. Drag and drop
CreateCSEntryChangeResult
activity into the Else branch of your IF activity and selectExportErrorCustomContinueRun
error code. Your Export Replace workflow looks like this:Drag and drop the Sequence activity in the empty Then branch of the first IF activity. Drag and drop WebSeviceCall activity inside the Sequence activity. Select SAPECC service name, ZSAPCONNECTORWS endpoint and BAPI_USER_CHANGE operation. Select on ... Arguments button to define parameters for web service call as follows:
Name | Direction | Type | Value |
---|---|---|---|
ADDRESS | In | BAPIADDR3 | address |
ADDRESSX | In | BAPIADDR3X | addressX |
COMPANY | In | BAPIUSCOMP | company |
COMPANYX | In | BAPIUSCOMX | company |
DEFAULTS | In | BAPIDEFAUL | defaults |
DEFAULTSX | In | BAPIDEFAX | defaultsX |
LOGONDATA | In | BAPILOGOND | logOnData |
LOGONDATAX | In | BAPILOGONX | logOnDataX |
RETURN | In/Out | TABLE_OF_BAPIRET2 | bapiret2Table |
USERNAME | In | String | userName |
- Select OK. The warning sign disappears. Your Export Replace workflow looks like this:
- To process change user request results, drag and drop the IF activity inside the Sequence activity after the WebServiceCall activity. Enter the following condition:
Not IsNothing(bapiret2Table.item) AndAlso bapiret2Table.item.Count(Function(errItem) errItem.TYPE.Equals("E") = True) <> 0
- If we get no errors, we assume that export operation completed successfully, and we want to indicate successful export of this object by creating
CSEntryChangeResult
with Success status. Drag and dropCreateCSEntryChangeResult
activity into Else branch of your IF activity and select Success error code. - Drag and drop Sequence activity into Then branch of your IF activity. Add Log activity with LogText value of
string.Join("\n",bapiret2Table.item.Where(Function(retItem) retItem.TYPE.Equals("E")).Select(Function(r) r.MESSAGE))
and Error tag. AddCreateCSEntryChangeResult
activity after Log activity with error code ofExportErrorCustomContinueRun
. Your Export Replace workflow looks like this:
You completed the definition of Export Replace workflow.
The next step is to configure ECMA2Host Webservice connector using this template.