Create an Integration
#
WelcomeThis tutorial guides you through creating an integration for Cortex XSOAR/Cortex XSIAM. You should have a working instance of Cortex XSOAR/XSIAM and programming experience with Python.
NOTE: As an introductory guide, not all code in the tutorial strictly follows our code conventions. See
Code Conventions to learn more about our coding standards.
The complete code for the integration is available at the end of the tutorial.
Although this tutorial is based on Cortex XSOAR, it can be used with the following platforms:
- Cortex XSOAR 6 and 8
- Cortex XSIAM
#
The Cortex XSOAR IDEYou have the option to develop integrations using the Cortex XSOAR/XSIAM IDE or a standalone IDE such as Visual Studio Code (if you use Visual Studio, see the Extension for Visual Studio Code). For this tutorial, use the Cortex XSOAR/XSIAM IDE, which includes access to the Script Helper (a library of many common server functions within Cortex XSOAR/XSIAM), and a graphical user interface for editing integration settings, commands, and arguments.
#
CommonServerPython & CommonServerUserPythonThe CommonServerPython (CSP) and CommonServerUserPython (CSUP) scripts are implicitly imported at the beginning of every Python script in Cortex XSOAR/XSIAM. CSP is imported first, enabling you to create common CSUP methods to use across scripts and integrations.
NOTE: CSP and CSUP scripts can’t be attached to integrations you create, so any changes you implement are not available for other users.
#
Script HelperIn many cases, there is an existing script for common server functions. With the Script Helper, you can easily find tools to format a table, manipulate data, post to the War Room, etc. If a function can be used in many different scripts, there’s a good chance it already exists in the Script Helper. If you create a new function that would be useful across many scripts, we encourage you to contribute that function to CommonServerPython scripts.
#
Navigate to BYOI (Bring Your Own Integration)In this example, you are going to create an English-to-Yoda translator, which translates normal English into Yoda (the Star Wars character) language. This is a simple integration that lets you explore important aspects of integration development. You can try calling an API, parsing data, and posting it to the War Room. You will use the Yoda-Speak translate API available at https://funtranslations.com/api/yoda.
Navigate to one of the following:
- Cortex XSOAR 8: Settings & Info > Settings > Integrations > Instances
- Cortex XSOAR 6: Settings > Integrations
- Cortex XSIAM: Settings > Configurations > Automation & Feed Integrations
Click BYOI in the top right corner.
If you don’t see this button, you don’t have the correct permissions required for creating new integrations. Ask your administrator for assistance.
#
Define Integration SettingsWhen clicking BYOI, you enter the Cortex XSOAR/XSIAM IDE and the HelloWorld integration template is loaded by default. You will replace the existing HelloWorld integration settings with the new Yoda Speak settings and delete any unused parameters or arguments.
The new settings are saved in a YAML file, which you can export and import. To learn more about integration YAML files, see Integrations and Scripts Metadata YAML File.
#
Basic Integration SettingsIn the BASIC section, add the following:
Parameter | Description |
---|---|
Integration Name | This integration will be called Yoda Speak. |
Description | The description usually includes basic information about the integration, common troubleshooting steps, and any required setup instructions. For the purposes of this tutorial, however, let's enter Creating an Integration, we are. |
Category | Select Utilties. For the full list of available categories, see Design Best Practices. |
Drag and Drop a file | (Logo in Cortex XSOAR 6) Add a logo, which should be no larger than 10KB, have a transparent background, and be in the PNG format. You can drag and drop the logo here or click the box to find the file for upload. |
NOTE:
For Cortex XSOAR, the Fetches incidents checkbox enables the integration to run periodically to ingest events and create incidents.
For Cortex XSIAM, the Fetches Alerts checkbox enables the integration to run periodically to ingest event and create alerts.
The Yoda Speak integration doesn't need to fetch incidents/alerts, so ensure this checkbox is empty. While you don’t need this feature for this Yoda Speak integration, fetching incidents/alerts is an essential part of many integrations. There are additional options available, but these are less commonly used:
NOTE: Mirroring is not supported in Cortex XSIAM.
#
ParametersThese parameters can be used across all commands in the integration. Some common parameters include the API key used for communicating with the product, your username, whether to use proxy, etc.
For the Yoda Speak integration, you want to include the API key and proxy settings, and to allow for insecure requests. You will also include a URL parameter, which tells the integration where to send requests. For each parameter, include a display name that tells the user why the value is used.
API Key
Add the following values for the API Key parameter:
API Key Parameter Settings | |
---|---|
Parameter name | apikey |
Type dropdown | Authentication |
Mandatory | not selected |
Display password | API Key |
In this integration, the free service does not require an API key, and allows up to 60 API calls a day with up to 5 calls an hour. If you require more API calls, FunTranslations offers a paid service that you can access with an API key. In this case, you provide the option for users with a paid subscription to enter their API key, but you don’t make it mandatory. The Authentication parameter type enables you to use Cortex XSOAR/XSIAM’s built-in credential management system to save and use the API key, and ensures that the API key is not displayed to the user and not stored in logs.
NOTE: Many integrations that connect to third party services require an API key for authentication, which is sent with every request to the third party service. Since it’s used by every command that performs an API call, you add it as a global parameter and not an argument.
URL
Add the following values for the URL parameter:
URL Parameter Settings | |
---|---|
Parameter name | url |
Type dropdown | Short Text |
Mandatory | selected ✓ |
Initial value | https://api.funtranslations.com/translate/ |
Display name | API URL |
NOTE: In some cases, the URL parameter may be used for third party services that allow you to connect to more than one server, such as servers in different geographic regions, for example https://login.example.de or https://login.example.com.
Insecure
Add the following values for the insecure parameter:
Insecure Parameter Settings | |
---|---|
Parameter name | insecure |
Type dropdown | Boolean |
Initial value | false |
Display name | Trust any certificate (not secure) |
When Trust any certificate is set to True, the integration ignores TLS/SSL certificate validation errors. Use this to test connection issues or connect to a service while ignoring SSL certificate validity. We do not recommend setting this option to true in a production environment.
Proxy
Add the following values for the proxy parameter:
Proxy Parameter Settings | |
---|---|
Parameter name | proxy |
Type dropdown | Boolean |
Initial value | false |
Display name | Use system proxy settings |
When Use system proxy settings is set to True, the integration runs using the proxy server (HTTP or HTTPS) defined in the server configuration. In most cases, a proxy is not required.
#
CommandsBefore you start coding, add commands in the Commands section.
- Click +Add command.
- For the Command name, enter yoda-speak-translate.
- For the description, enter Translates a text from English to Yoda.
NOTE: Command names should follow “brand-function” name formatting convention. For example, VirusTotal uses the vt-comments-add command that adds a comment to a scan. An exception to this rule is that when creating commands that enrich indicators, the commands should be named according to the indicator: !ip, !domain, etc. This naming convention allows commands from multiple integrations to be run together to enrich an indicator. For example, running !ip ip=8.8.8.8 can trigger multiple integrations that gather information about the IP address. For more information, see Generic Reputation Commands.
Arguments
You want to translate English text to Yoda-style text, so you need to add an argument called text. Users can provide a different text string on every call. The argument is mandatory, since the command can’t run if there’s nothing to translate.
NOTE: Unlike parameters, arguments are specific to each command.
Outputs
In Cortex XSOAR/XSIAM, the context is a JSON object that is created for each incident and stores results from integration commands and automation scripts. Context is important because it enables you to add information to an incident and run playbooks and integrations utilizing that information.
To write the translation to context, you can add an output to the yoda-speak-translate command. The naming convention for the context path is Brandname.Object.Property, so you will add YodaSpeak.Phrase.Translation as the context path. For a description, enter Translation text, in Yodish. Select type String.
#
Integration CodeOnce you have finished adding your parameters, command, argument, and outputs, you can write the integration code.
NOTE: The sample code uses standard Python error handling mechanisms, such as try. For more information about errors and exceptions in Python, see the Python documentation. In this integration code, you want to raise exceptions when errors occur. The convention is to have a main try/except block on main(), that catches errors and calls return_error.
The return_error function ensures that playbooks calling these functions will fail and stop, alerting the user to a problem. In integrations and automations, you refrain from calling return_error in other places in the code.
#
ImportAt the beginning of the code, you can import Python libraries, so that its commands are available for our integration. Every integration runs inside a Docker image, and our standard Docker image includes most of the common packages, such as JSON and collections. In our Yoda Speak integration, you don’t need to import any libraries, as it only uses the BaseClient class, implicitly imported from CommonServerPython.
NOTE: When working within a traditional IDE, such as PyCharm or Visual Studio Code, we recommend importing the following at the top of your code for debugging purposes.
For Cortex XSOAR, if you want to use Python libraries that are not included in the standard Cortex XSOAR Docker image, you can create a customized Docker image. For Cortex XSOAR 8, see Change the Docker Image.
#
Disable Secure WarningsNext you want to prevent Python from raising a warning when accessing resources insecurely.
Since you created the insecure parameter that allows the integration to ignore TLS/SSL certificate validation errors, you also need to disable the warning.
#
Create the Class ClientThe Client is an object that communicates with the API. You create a class called Client. When a Client object is created, it instantiates a parent BaseClient using the parameters you have set up (whether to use proxy, whether to allow insecure connections, and the base URL). If the user provides values to the api_key parameter, the Client sets the headers it will use accordingly.
NOTE: When using the Yoda Speak API with an API key, the API key is passed as a header.
The number of methods our Client class has usually matches the number of commands in our integration. The Yoda Speak integration only has the translation command, so our Client object should have a matching method to the API request which returns its result.
#
Create the test_moduleThe test_module function is run whenever the Test integration button is clicked in the integration instance settings. The test_module function sends a hardcoded preset string (here, it’s I have the high ground) to the Yoda-Speak translate API to test API connectivity and authentication. There are three possible results:
- HTTP response code is 200, which means the request is successful. You return the string ok per the convention for a successful test.
- The request is not successful and the problem is related to authorization: Authorization Error: make sure API Key is correctly set.
- The request is not successful for any other reason: The error text is displayed.
#
Create the Translate CommandThe translate_command function uses the client that is provided as an argument for the function and it calls translate using the text provided. The client is created outside of the function (in main()). The function performs several steps.
Confirms that there is a non-empty string to translate. If the string input is empty, it raises an exception.
Tells the Client to send the appropriate API call. If the translation fails (due to an API rate limit, authentication or connection error, etc.), an exception is raised.
If the translation succeeds, you want to return it to Cortex XSOAR/XSIAM. To do this, use a class called CommandResult (which is declared in CSP). You supply it with the following arguments:
outputs: You create a dictionary called outputs where both the original text and the translation are stored.
outputs_prefix: The first level of the output in the context data. It usually matches the name of the integration or service.
raw_response: The argument used to attach the raw response received from the service, which can be useful when debugging unexpected behaviors.
outputs_key_field: Since you can run the translation command multiple times, and possibly receive different results for the same string of text, the system needs to know where to update or append each result. In this example you tell the system that Phrase.Original is the key that represents the original text you translated, so that the next time the command is run on the same string of text, the translated values will update. Learn more about storing results as context data.
readable_output: This is what users see in their War Room when calling the command, so it should be formatted. You can use the tableToMarkdown function (from CSP) to turn the JSON into a user-friendly table. You provide tableToMarkdown with both the JSON values and a title for the table.
NOTE: The Script Helper provides an easy way to insert common functions into your code. If you click the Script Helper button and search for the tableToMarkdown command, you have the option to insert it directly into the code with placeholders for its name (title) and t (JSON) arguments.
#
Create the Main FunctionEverything actually runs within Main. You pull in the integration parameters, arguments, and the translate command. The parameters are assigned to variables. Notice that the parameters are the same ones you set up in the integration settings earlier.
When the function runs, the command will be logged for debugging purposes.
You now create a Client using the given parameters. The Client is defined.
There are two possible commands that can be passed to the main function in our integration.
test-module If the command name is test-module, it means the user has clicked the integration test button while setting up or editing an integration instance. NOTE: You did not explicitly create a command called test-module. It is a built-in command.
When returning **ok**, the user is shown a green **Success** message. If any value other than **ok** is returned, an error is displayed. Ensure you return errors that help the user understand what to change in the integration settings to fix connection issues.yoda-speak-translate This is the primary command for our integration and lets us translate strings of text.
There is also an else option. This returns an error if someone tries to run a command that was created in the YAML file but does not exist in the Python (PY) file. For example, if you added a command yoda-interpret in the integration settings, but did not add it to this file, and then tried to run that command, you would see yoda-interpret is not implemented.
#
Log ErrorsIf any errors occur during the execution of our code, show those errors to the user and also return an error.
NOTE: demisto.debug must be used to print debug logs. Using a print statement can cause unexpected exceptions and errors.
#
Start at MainThis line tells the system where to start running your code. By convention, you call the main function main.
#
Test the IntegrationPlatform | Navigation |
---|---|
Cortex XSOAR 8 | Settings & Info > Settings > Integrations > Instances > Yoda > Add Instance |
Cortex XSOAR 6 | Settings > Integrations > Yoda > Add Instance |
Cortex XSIAM | Settings > Configurations > Automations and Feed Integrations > Yoda > Add Instance |
You don't need to enter an API key, but will instead use the free option with a limited number of API calls. To test connectivity, click Test. If the connection is successful, you will see Success and the date/time displayed. Click Save & Exit.
NOTE: If you have an integration open in two different tabs, you may encounter an error where your changes aren’t saved. In this case, take a screenshot of your changes, close both tabs, and then reopen one tab. Enter your changes again and save.
To test the integration, in the CLI, enter !yoda-speak-translate and the text argument with the value Hello, my name is John Smith. We are learning about integrations.
NOTE: In Cortex XSIAM, to access the CLI, select Incident Response > Automation > Playground
In the Playground, you can see the table you created with the tableToMarkdown function, with the results.
View the same translation in the context.
YodaSpeak is the root for Phrase. If the translation changes the next time you run the command, the translation field will be updated.
#
Create a PlaybookYou can see the real power of integrations when you include them in a playbook. Go to the Playbooks page and click New Playbook.
NOTE: In Cortex XSIAM, go to Incident Response > Automation > Playbooks.
Name the playbook Yoda Speak.
The playbook translates the Details field in an incident into Yoda Speak.
In the task library, search for yoda and click Add.
You can see there is a field for text, which is a required argument. While you could type text here, instead you want to pull the string from the incident Details field.
Do one of the following:
- For Cortex XSOAR, click the curly brackets, Incident details, Details, and click OK.
- For Cortex XSIAM, click the curly brackets, Alert details, Details, and click OK.
For Cortex XSOAR 8:
For Cortex XSIAM:
To print the translation to the War Room, add a print task.
In the task library, search for print and under utilities, add the Print task.
In the value field, as you want to pull our text from the incident, click the curly brackets.
Our options now include yoda-speak-translate.
Under yoda-speak-translate, select Translation and click OK.
Connect the tasks in our playbook. Use your cursor to create lines between all tasks.
Go back to the Playbook Settings > Advanced, and deselect Quiet Mode.
This ensures that you see that results are printed to the War Room.
Save the playbook.
#
Create an IncidentTo test our new playbook, go to the Incidents page and create a new incident. In the Details field, enter the text for translation: The prequel movies are more entertaining than the new Disney movies. Assign the Yoda Speak playbook, and click Create new incident.
For Cortex XSOAR:
For Cortex XSIAM:
Select the incident you just created. Go to the Work Plan page and you can see that your playbook executed successfully. Go to the War Room and view the translation.
#
We’re Done!Your example integration is now complete, and you can use it throughout Cortex XSOAR/XSIAM.
Real world integrations are usually more complex than our example. Like any code, integrations require maintenance and can be extended over time with new features, commands etc.
To ensure integrations perform as expected, content packs can have unit tests, and test playbooks. Learn more about contributing content.
#
ResourcesThe YodaSpeak content pack is available on GitHub. The pack contains the standard pack file structure and can be viewed as a resource when creating new integrations.