Use SAP function calling with Gemini

This document provides an overview of how to use SAP function calling with Gemini, by using the Vertex AI SDK for ABAP in your SAP BTP, ABAP environment.

You can define custom functions and provide them to the Gemini models by using the Function Calling feature. The models do not directly invoke the custom functions, but instead generate structured data output that specifies the function name and suggested arguments. This output lets you write applications that take the structured output and call external APIs. The resulting API output can then be incorporated into a further model prompt, allowing for more comprehensive query responses.

The Vertex AI SDK for ABAP simplifies function calling for ABAP developers by providing them opportunities to invoke custom logic written in SAP function modules by:

  • Passing SAP function module name or OData (Open Data Protocol) service name to the model as function declarations, describing the name of the function or service, their purpose, and related parameters.
  • Setting auto-invocation of SAP function module or OData service implicitly while invoking the model.

Before you begin

Before using the Vertex AI SDK for ABAP for SAP function calling with Gemini, make sure that you or your administrators have completed the following prerequisites:

Enrich the Gemini AI model context with SAP data by invoking an SAP function module

This section explains how you can enrich the Gemini AI model context with SAP data by invoking an SAP function module.

The following is a typical ABAP developer's journey to invoke an SAP function module:

  1. User provides an input prompt.
  2. The SDK passes the input prompt and function declarations to the model.
  3. The model reviews the prompt and declared functions to derive the function to call and suggests the parameter values to call the function.
  4. If auto invocation is set, then the SDK calls the SAP function module.
  5. The SDK then invokes the model with the output of the called function.
  6. The model responds with a reliable answer for the final prompt enriched with the output of the called function.
  7. The SDK returns the response to the user.

If you choose not to auto-invoke SAP function modules, then the SDK lets you use the function calling feature without any SAP function module invocation. In this case, you can follow the typical function calling workflow to use external tools such as APIs and functions.

SAP function calling with Gemini in an SAP BTP, ABAP environment

Instantiate the Gemini multimodal invoker class

To invoke function calling in SAP, you use the /GOOG/CL_GENERATIVE_MODEL class. You instantiate the class by passing the model key configured in the model generation parameters.

DATA(lo_model) = NEW /goog/cl_generative_model( iv_model_key = 'MODEL_KEY' ).

Replace MODEL_KEY with the model key name, which is configured in the model generation parameters.

Create an SAP function module

To create an SAP function module for auto-invocation by the SDK, follow the provided schema:

Category Parameter name Associated type
Importing IT_FUNCTION_PARAMETERS /GOOG/T_FUNCTION_PARAMETERS
Exporting EV_FUNCTION_RESPONSE STRING
Changing CV_PROMPT STRING
Exceptions /GOOG/CX_SDK Exception Class

Based on the importing parameters, write your custom logic within the function module, which can be either fetching SAP data through SELECT queries, or calling an external API or module to get the missing information.

Set the exporting parameter EV_FUNCTION_RESPONSE with the information to feedback to the LLM context. You can also change or modify the prompt text in CV_PROMPT based on the custom logic and your business requirement to further instruct the LLM based on different business scenarios.

Add function declaration

To add a function declaration to the LLM context, you can use the ADD_FUNCTION_DECLARATION method. Call the ADD_FUNCTION_DECLARATION method each time you need to add a function to the context.

DATA lt_parameters TYPE /goog/cl_generative_model=>tt_parameter_properties.
APPEND VALUE #( parameter_name = 'PARAMETER_NAME'
                type           = 'PARAMETER_TYPE'
                description    = 'PARAMETER_DESCRIPTION'
                is_required    = 'PARAMETER_IS_REQUIRED' ) TO lt_parameters.
lo_model->add_function_declaration( iv_name        = 'FUNCTION_MODULE_NAME'
                                    iv_description = 'FUNCTION_MODULE_DESCRIPTION'
                                    it_parameters  = lt_parameters ).

Replace the following:

  • PARAMETER_NAME: Name of the parameter.
  • PARAMETER_TYPE: The data type of the parameter, such as string, integer, or boolean.
  • PARAMETER_DESCRIPTION: A clear explanation of the parameter's purpose and expected format.
  • PARAMETER_IS_REQUIRED: If this parameter is mandatory for the function to operate, then set the value to ABAP_TRUE.
  • FUNCTION_MODULE_NAME: Name of the SAP function module.
  • FUNCTION_MODULE_DESCRIPTION: Description of the SAP function module.

You can also declare a function without parameters, which can serve as a fallback or backup function. If a user prompt doesn't provide enough information to call a specific function, you can instruct Gemini to select this fallback function.

Set auto invocation of SAP function module

To set auto invocation of the selected SAP function that is selected by the model, you can use the SET_AUTO_INVOKE_SAP_FUNCTION method. If ABAP_TRUE is passed in the importing parameter IV_AUTO_INVOKE, then the function module is invoked by the SDK and its response is included with the LLM context to generate the final response.

You must define your function module by following the schema described in the section Create SAP function module.

lo_model->set_auto_invoke_sap_function( abap_true ).

Generate content with function calling

To pass the prompt text to the Gemini model, you can use the GENERATE_CONTENT method. To get the response generated by Gemini with the additional context added from the SAP function module through function calling, use the GET_TEXT method.

DATA(lv_response) = lo_model->generate_content( iv_prompt_text ='PROMPT'
                           )->get_text( ).

Replace PROMPT with your text prompt.

Get the selected function name and parameter values

To get the function selected by Gemini (from among the declared functions) and its suggested parameters, use the GET_FUNCTION_CALL method.

DATA(lo_response) = lo_model_key->set_auto_invoke_sap_function( abap_true
                               )->generate_content( iv_prompt_text = 'PROMPT' ).
lo_response->get_function_call( IMPORTING ev_function_name       = DATA(lv_function_name)
                                          et_function_parameters = DATA(lt_function_parameters) ).

Replace PROMPT with your text prompt.

You can get the name of Gemini's selected function from LV_FUNCTION_NAME and the suggested parameters from LT_FUNCTION_PARAMETERS. You can use this information to validate, track, and log Gemini actions as per your enterprise's security information and event management guidelines.

Code sample

The following code sample illustrates how to use SAP function calling to receive a final response from the model.

DATA lt_parameters TYPE /goog/cl_generative_model=>tt_parameter_properties.
TRY.
  DATA(lo_model) = NEW /goog/cl_generative_model( iv_model_key = 'MODEL_KEY' ).
  APPEND VALUE #( parameter_name = 'PARAMETER_NAME'
                  type           = 'PARAMETER_TYPE'
                  description    = 'PARAMETER_DESCRIPTION'
                  is_required    = 'PARAMETER_IS_REQUIRED' ) TO lt_parameters.
  DATA(lv_response) = lo_model->add_function_declaration(
                                    iv_name        = 'FUNCTION_MODULE_NAME'
                                    iv_description = 'FUNCTION_MODULE_DESCRIPTION'
                                    it_parameters  = lt_parameters
                              )->set_auto_invoke_sap_function( abap_true
                              )->generate_content( iv_prompt_text ='PROMPT'
                              )->get_text( ).
  IF lv_response IS NOT INITIAL.
      cl_demo_output=>display( lv_response ).

  ENDIF.
 CATCH /goog/cx_sdk INTO DATA(lo_cx_sdk).
  cl_demo_output=>display( lo_cx_sdk->get_text( ) ).

ENDTRY.

Replace the following:

  • MODEL_KEY: The model key name, which is configured in the model generation parameters.
  • PARAMETER_NAME: Name of the parameter.
  • PARAMETER_TYPE: The data type of the parameter, such as string, integer, or boolean.
  • PARAMETER_DESCRIPTION: A clear explanation of the parameter's purpose and expected format.
  • PARAMETER_IS_REQUIRED: If this parameter is mandatory for the function to operate, then set the value to ABAP_TRUE.
  • FUNCTION_MODULE_NAME: Name of the SAP function module.
  • FUNCTION_MODULE_DESCRIPTION: Description of the SAP function module.
  • PROMPT: Your text prompt.

Enrich the Gemini AI model context with SAP data by invoking an OData service

This section explains how you can enrich the Gemini AI model context with SAP data by invoking an OData service.

The following is a typical ABAP developer's journey to invoke an OData service:

  1. User provides an input prompt.
  2. The SDK passes the input prompt and function declarations to the model.
  3. The model reviews the prompt and declared functions to derive the function to call and suggests the parameter values to call the function.
  4. If auto invocation is set, then the SDK calls the OData service.
  5. The SDK then invokes the model with the output of the called OData service.
  6. The model responds with a reliable answer for the final prompt enriched with the output of the called OData service.
  7. The SDK returns the response to the user.

If you choose not to auto-invoke OData services, then the SDK lets you use the function calling feature without any OData service invocation. In this case, you can follow the typical function calling workflow to use external tools such as APIs and functions.

SAP function calling with Gemini in an SAP BTP, ABAP environment

Create an OData service

To create an OData service for auto-invocation by the SDK as a function, use the following schema:

{
    "UserRequest": "Input Prompt as user request",
    "FunctionResponse": null,
    "ModifiedUserRequest": null,
    "ContextParameters": [
        {
            "ParameterName": "Parameter 1",
            "ParameterValue": "Value 1"
        },
        {
            "ParameterName": "Parameter 2",
            "ParameterValue": "Value 2"
        }
    ]
}

The following table provides information about the fields used in this schema:

Schema field Field type Input/Output mode Description
UserRequest String Input only The input prompt that the user enters.
FunctionResponse String Output only The output of the OData service. It can be a single value or a serialized JSON string.
ModifiedUserRequest String Output only The modified UserRequest, which is fed to Gemini as the prompt.
ContextParameters Table of parameters Input only The table of parameters, which includes parameter names and values.
ParameterName String Input only Name of the function parameter.
ParameterValue String Input only Value of the function parameter, as suggested by Gemini.

Create the OData service entities

  1. Create an entity by using the preceding schema.

    This entity accepts the user request as a prompt, forwards it to Gemini, and is declared as a function by using Vertex AI SDK for ABAP. This entity also carries back the OData function response from the backend SAP system to BTP, which is then augmented to the Gemini AI model context.

    1. Set an appropriate name for the new item.
    2. Set values for the following attributes. Retain the default values for all other attributes.

      Name Is key? EDM core type ABAP field name
      UserRequest Yes Edm.String USER_REQUEST
      FunctionResponse No Edm.String FUNCTION_RESPONSE
      ModifiedUserRequest No Edm.String MODIFIED_USER_REQUEST

    If you want to place more entities in the same OData service, then create them by using the same schema.

  2. Create another entity named ContextParameters with the following attributes. Retain the default values for all other attributes.

    This entity contains the function parameters and their values that the Gemini model provides.

    Name Is key? EDM core type ABAP field name
    ParameterName Yes Edm.String PARAMETER_NAME
    ParameterValue Yes Edm.String PARAMETER_VALUE
  3. Create entity sets for all the entities that you created as part of the preceding steps.

  4. Create an association between each entity that you created in step 1 with the ContextParameters entity that you created in step 2 as follows:

    1. Set a name for the association.
    2. In the Principal Entity field, set the name of the entity that you created in step 1.
    3. In the Principal Entity Cardinality field, set M.
    4. In the Dependent Entity field, set ContextParameters.
    5. In the Dependent Entity Cardinality field, set M.
    6. Save and generate the OData runtime artifacts.
  5. Add the created OData service in SAP Gateway and activate the ICF node.

  6. Generate the EDMX file for the OData metadata as follows:

    1. Go to TCode /iwfnd/maint_services and get the metadata for the OData service.
    2. Format and save the generated XML with the file extension .edmx.

Write the OData function logic

  1. In the generated MPC_EXT class, add and activate a deep entity type:

    TYPES:
      BEGIN OF ts_function_deep_entity,
        user_request          TYPE string,
        function_response     TYPE string,
        modified_user_request TYPE string,
        context_parameters    TYPE TABLE OF ts_contextparameters WITH DEFAULT KEY,
    
      END OF ts_function_deep_entity.
    
  2. Redefine the CREATE_DEEP_ENTITY method of the generated DPC_EXT class by completing the following steps:

    1. If you created more than one OData entity in the Create the OData service entities section, then add a CASE statement for each OData entity.
    2. Write the ABAP logic to get or prepare the response to augment the context for Gemini in the BTP application.
    3. In the FUNCTION_RESPONSE field, set the response.
    4. In the MODIFIED_USER_REQUEST field, set the modified prompt.
    5. By using the COPY_DATA_TO_REF OData method, bind the overall response in the message.

    The following snippet is a sample ABAP logic:

    TRY.
         DATA: ls_request TYPE <z*_mpc_ext>=>ts_function_deep_entity.
    
         CASE io_tech_request_context->get_entity_type_name( ).
           WHEN <z*_mpc_ext>=>gc_ENTITY_NAME.
             CALL METHOD io_data_provider->read_entry_data
               IMPORTING
                 es_data = ls_request.
    
     <ABAP logic to formulate OData function response>
    
             ls_request-modified_user_request = 'MODIFIED_GEMINI_PROMPT'.
             ls_request-function_response = 'ODATA_FUNCTION_RESPONSE'.
    
             CALL METHOD me->copy_data_to_ref
               EXPORTING
                 is_data = ls_request
               CHANGING
                 cr_data = er_deep_entity.
     ENDCASE.
         CATCH /iwbep/cx_mgw_busi_exception.
         CATCH /iwbep/cx_mgw_tech_exception.
     ENDTRY.
     ```
    

Replace the following:

  • ENTITY_TYPE: the name of the OData entity that you created in step 1 of the Create the OData service entities section
  • MODIFIED_GEMINI_PROMPT: the modified prompt provided by Gemini
  • ODATA_FUNCTION_RESPONSE: the response provided by the OData service

If you have more than one field, or a nested structure or table to be passed as the function response, then you can serialize the ABAP type into a string by using the SERIALIZE_JSON method of the /GOOG/CL_JSON_UTIL class of the SDK.

Define the OData service as a service consumption model on SAP BTP

After you create the OData service, you need to define it as a service consumption model in your SAP BTP environment. You do this by creating a service consumption model, an HTTP outbound service, a communication scenario, and a communication arrangement.

Create a service consumption model

  1. In your ABAP project, open the context menu and select New > Other ABAP Repository Object > Business Services > Service Consumption Model.
  2. In the New Service Consumption Model window, complete the following steps:
    1. In the Package field, select and upload the EDMX file that you generated earlier in an earlier section.
    2. Enter a name and description.
    3. In the Remote Consumption Mode field, select OData.
    4. Click Finish.

Create an HTTP outbound service

  1. In your ABAP project, open the context menu.
  2. Select your package and click New > Other ABAP Repository Object.
  3. In the Service Type field, select HTTP Service.
  4. In the Default Path Prefix field, enter the path prefix for your OData service in the backend SAP system.

    You can find out the OData path prefix by accessing the SAP Gateway Client. The value you see in the Request URI field is the path prefix for your OData service.

  5. Click Next.

Create a communication scenario

After you create an HTTP outbound service, you must create a communication scenario and assign the outbound service to it. To do so, complete the following steps:

  1. In your ABAP project, select Communication Scenario and then click Next.
  2. In the New Communication Scenario window, complete the following steps:
    1. Enter a name and description.
    2. Click Next.
    3. In the Communication Scenario Type field, select Customer Managed.
    4. Go to the Outbound tab.
    5. In the Allowed Instances field, select One instance per scenario & communication system.
    6. In the Supported Authentication methods field, select the following checkboxes: Basic, X.509, and OAuth 2.0.
    7. In the OAuth 2.0 Grant Type field, select SAML 2.0 Bearer Assertion.
    8. Under the Outbound Services section, click Add and then select the outbound service that you created in the previous section.
    9. In the Outbound Service section, make sure that the Default Path Prefix is correct. If it's not, then click Synchronize.
    10. Save the communication scenario.

Create a communication arrangement

  1. Access the SAP Fiori launchpad of the BTP ABAP system where the ABAP SDK for Google Cloud is installed.
  2. Open the Communication Arrangement app.

    1. Click New.
    2. In the New Communication Arrangement dialog that appears, enter a value for the following fields:

      • Scenario: select the communication scenario that you created in the Create a communication scenario section.
      • Arrangement name: enter a name for the communication arrangement.
    3. Click Create.

    4. For the communication arrangement that appears, in the Communication System field, select the communication system that you created in the previous step.

    5. Click Save.

Instantiate the Gemini multimodal invoker class

To invoke function calling in SAP, you use the /GOOG/CL_GENERATIVE_MODEL class. You instantiate the class by passing the model key configured in the model generation parameters.

DATA(lo_model) = NEW /goog/cl_generative_model( iv_model_key = 'MODEL_KEY' ).

Replace MODEL_KEY with the model key name, which is configured in the model generation parameters.

Add function declaration

To add an OData service as a function declaration to the LLM context, you can use the ADD_FUNCTION_DECLARATION method. Call the ADD_FUNCTION_DECLARATION method each time you need to add an OData service as a function to the context.

DATA lt_parameters TYPE /goog/cl_generative_model=>tt_parameter_properties.
DATA ls_connection_parameters TYPE /goog/cl_generative_model=>ty_connection_parameters.

APPEND VALUE #( parameter_name = 'PARAMETER_NAME'
                type           = 'PARAMETER_TYPE'
                description    = 'PARAMETER_DESCRIPTION'
                is_required    = 'PARAMETER_IS_REQUIRED' ) TO lt_parameters.

ls_connection_parameters-connection_type       = 'ODATA'.
ls_connection_parameters-comm_scenario         = 'COMMUNICATION_SCENARIO'.
ls_connection_parameters-service_id            = 'ODATA_SERVICE_ID'.
ls_connection_parameters-proxy_model_id        = 'ODATA_SERVICE_CONSUMPTION_MODEL_ID'.
ls_connection_parameters-relative_service_root = 'ODATA_ROOT_URI'.
ls_connection_parameters-entity_set_name       = 'ODATA_ENTITY_SET_NAME'.

lo_model->add_function_declaration( iv_name                  = 'FUNCTION_NAME'
                                    iv_description           = 'FUNCTION_DESCRIPTION'
                                    it_parameters            = lt_parameters
                                    is_connection_parameters = ls_connection_parameters ).

Replace the following:

  • PARAMETER_NAME: the name of the parameter
  • PARAMETER_TYPE: the data type of the parameter, such as string, integer, or boolean
  • PARAMETER_DESCRIPTION: a clear explanation of the parameter's purpose and expected format
  • PARAMETER_IS_REQUIRED: set the value to ABAP_TRUE if this parameter is mandatory for the function to operate
  • COMMUNICATION_SCENARIO: the ID of the communication scenario that you created
  • ODATA_SERVICE_ID: the ID of the outbound service that you created for the OData service
  • ODATA_SERVICE_CONSUMPTION_MODEL_ID: the ID of the service consumption model that you created for the OData service
  • ODATA_ROOT_URI: the root URI of the OData service in the backend SAP system
  • ODATA_ENTITY_SET_NAME: the entity set name of the OData service that is to be declared as a function
  • FUNCTION_NAME: the name of the function that you declared for the OData service you want to use
  • FUNCTION_DESCRIPTION: the description of the function that you declared for the OData service you want to use

Set auto invocation of OData service

To set auto invocation of the SAP function OData service that is selected by the Gemini model, you can use the SET_AUTO_INVOKE_SAP_FUNCTION method. If ABAP_TRUE is passed in the importing parameter IV_AUTO_INVOKE, then the function module is invoked by the SDK and its response is included with the LLM context to generate the final response.

You must define your function module by following the schema described in the section Create SAP function module.

lo_model->set_auto_invoke_sap_function( abap_true ).

Generate content with function calling

To pass the prompt text to the Gemini model, you can use the GENERATE_CONTENT method. To get the response generated by Gemini with the additional context added from the SAP function module through function calling, use the GET_TEXT method.

DATA(lv_response) = lo_model->generate_content( iv_prompt_text ='PROMPT'
                           )->get_text( ).

Replace PROMPT with your text prompt.

Get the selected function name and parameter values

To get the function selected by Gemini (from among the declared functions) and its suggested parameters, use the GET_FUNCTION_CALL method.

DATA(lo_response) = lo_model_key->set_auto_invoke_sap_function( abap_true
                               )->generate_content( iv_prompt_text = 'PROMPT' ).
lo_response->get_function_call( IMPORTING ev_function_name       = DATA(lv_function_name)
                                          et_function_parameters = DATA(lt_function_parameters) ).

Code sample

The following code sample illustrates how to use SAP function calling with an Odata service to receive a final response from the model:

DATA lt_parameters TYPE /goog/cl_generative_model=>tt_parameter_properties.
DATA ls_connection_parameters TYPE /goog/cl_generative_model=>ty_connection_parameters.
TRY.
     DATA(lo_model) = NEW /goog/cl_generative_model( iv_model_key = 'MODEL_KEY' ).
     APPEND VALUE #( parameter_name = 'PARAMETER_NAME'
                     type           = 'PARAMETER_TYPE'
                     description    = 'PARAMETER_DESCRIPTION'
                     is_required    = 'PARAMETER_IS_REQUIRED' ) TO lt_parameters.

     ls_connection_parameters-connection_type       = 'ODATA'.
     ls_connection_parameters-comm_scenario         = 'COMMUNICATION_SCENARIO'.
     ls_connection_parameters-service_id            = 'ODATA_SERVICE_ID'.
     ls_connection_parameters-proxy_model_id        = 'ODATA_SERVICE_CONSUMPTION_MODEL_ID'.
     ls_connection_parameters-relative_service_root = 'ODATA_ROOT_URI'.
     ls_connection_parameters-entity_set_name       = 'ODATA_ENTITY_SET_NAME'.

     DATA(lv_response) = lo_model_key->add_function_declaration(
                                         iv_name                  = 'FUNCTION_NAME'
                                         iv_description           = 'FUNCTION_DESCRIPTION'
                                         it_parameters            = lt_parameters
                                         is_connection_parameters = ls_connection_parameters
                                   )->set_auto_invoke_sap_function( abap_true
                                   )->generate_content( iv_prompt_text ='PROMPT'
                          )->get_text( ).
        IF lv_response IS NOT INITIAL.
          out->write( lv_response ).

      ENDIF.

  CATCH /goog/cx_sdk INTO DATA(lo_cx_sdk).
      out->write( lo_cx_sdk->get_text( ) ).
ENDTRY.

What's next