Overview
This module enables you to be notified of new Virto Commerce events or changes via the message queue of your choice.
The module is used to trigger an asynchronous background process in response to an event on the Virto Commerce platform.
As a payload, a Virto Commerce event delivers one of the predefined messages or any change to a resource.
This enables event-driven, reactive programming, which uses a publish-subscribe model. Publishers emit events but have no expectation about which events are handled. Subscribers decide which events they want to handle.
Key features
- Notification on new events from any module
- Supporting multiple destination providers
- Supporting custom destination providers (contact us if you need a new destination)
- Configurable via API as well as through application configuration (
appsettings.json
, environment variables, etc.) - Additional event filtering with the
JsonPath
expression - Preprocessing event data with Scriban-template enables fine-tuning the payload for the destination provider
- High performance
- Predefined destination provider: Azure Event Grid with CloudEvents-based data format
Example Uses for Event Bus
Serverless Application Architectures
Event Bus connects Virto Commerce and event handlers. For example, use Azure Event Grid to instantly trigger a serverless function to run currency exchange each time a new price is added to a price list.
Approval Process Automation
Event Bus allows you to speed up automation and simplify the approval process enforcement. For example, Azure Event Grid can notify Azure Automation when a new order is created or a new customer is registered. These events can be used to automatically check the entity configurations are compliant, put metadata, change status, or send an email notification.
Application Integration
Event Bus connects your app with other services. For example, you can create an application topic to send Virto Commerce event data to Event Grid and take advantage of its reliable delivery, advanced routing, and direct integration with Azure.
Alternatively, you can use Event Grid with Logic Apps to process data anywhere, without writing any code.
Configuring Event Translation
There are two ways to configure event translation within the Event Bus module: * Configuration options * API endpoints
Both of them share equal options data structures.
Note
In order to access API endpoints, you will need to create an API Key and grant permission before the call.
Configuring Provider Connections
Each provider connection definition is a link between provider type and connection options. You can have multiple connections to various destinations through a single provider.
Provider Connection Options Data Structure
Example:
{
"Name": "AzureEventGrid cloud",
"ProviderName": "AzureEventGrid",
"ConnectionOptionsSerialized": "{\"ConnectionString\": \"https://*.*.eventgrid.azure.net/api/events\", \"AccessKey\": \"kvpXffggvvMiNjBeBKdroX1r45UvZloXMwlM7i1TyqoiI=\"}"
}
Description:
Name | Description |
---|---|
Name | Human-readable connection name to distinguish the connection. Should be unique in the configuration |
ProviderName | Predefined destination provider name. Unique for each type of providers. Refers to the name of desired provider |
ConnectionOptionsSerialized | Provider-specific connection options |
Managing Provider Connections through Configuration
Add connections array under the EventBus:Connections
key. Connections configured in such a manner cannot be removed or updated through REST API.
If you have connections with the same name in the DB and configuration options, the one specified in the configuration options will be preferred.
Managing Provider Connections through REST API
Adding New Connection
Endpoint: /api/eventbus/connections
Method: POST
Request:
{
"name": "string",
"providerName": "string",
"connectionOptionsSerialized": "string"
}
Removing Connection by Name
Endpoint: /api/eventbus/connections/{name}
Method: DELETE
Request parameter: Provider connection name
Updating connection
Endpoint: /api/eventbus/connections
Method: PUT
Request:
{
"name": "string",
"providerName": "string",
"connectionOptionsSerialized": "string"
}
Getting Connection by Name
Endpoint: /api/eventbus/connections/{name}
Method: GET
Request parameter: Provider connection name
Response:
{
"name": "string", // Connection name
"providerName": "string", // Provider name
"connectionOptionsSerialized": "string", // Provider-specific connection options
"createdDate": "2022-08-26T13:52:55.932Z",
"modifiedDate": "2022-08-26T13:52:55.932Z",
"createdBy": "string", // If null, the connection specified in the configuration
"modifiedBy": "string", // If null, the connection specified in the configuration
"id": "string" // If null, the connection specified in the configuration
}
Searching for Connections
Endpoint: /api/eventbus/connections/{name}
Method: GET
Request:
{
"name": "string", // Connection name (optional, pass to search by name)
"providerName": "string", // Provider name (optional, pass to search by provider name)
"skip": 0, // providers to skip in paged loading
"take": 0 // providers to take in paged loading
}
Response:
{
"totalCount": 0,
"results": [
{
"name": "string", // Connection name
"providerName": "string", // Provider name
"connectionOptionsSerialized": "string", // Provider-specific connection options
"createdDate": "2022-08-26T13:52:55.932Z",
"modifiedDate": "2022-08-26T13:52:55.932Z",
"createdBy": "string", // If null, the connection specified in the configuration
"modifiedBy": "string", // If null, the connection specified in the configuration
"id": "string" // If null, the connection specified in the configuration
}
]
}
Configuring Subscriptions
Subscription is a rule that specifies which events should be caught and forwarded to a selected provider connection. You can also translate the event body to fit the payload needs of a provider.
Subscription Options Data Structure
Example:
{
"ConnectionName": "AzureEventGrid",
"Name": "Eventgrid forwarder",
"JsonPathFilter": "$",
"PayloadTransformationTemplate": "",
"EventSettingsSerialized": null,
"Events": [
{
"EventId": "VirtoCommerce.YourModule.Web.Events.NewCompanyRegistrationRequestEvent"
}
]
}
Description:
Name | Description |
---|---|
ConnectionName | Name of the connection the event data should be forwarded to. |
Name | Human-readable name to distinguish subscriptions. Should be unique in configuration |
JsonPathFilter | JsonPath filter expression that allows you to additionally filter events that have specific value in the body. If the body with applied JsonPath filter does not yield any value, the module will not call the provider. The default value is $ , which means any event body is OK and may be transferred to the provider. We use Newtonsoft.Json to query Jsons. Find out more about JsonPath here. Also there is an useful tool to test your jsons and queries. |
PayloadTransformationTemplate | An optional setting where you can specify a Scriban-template to transform event data to a different form. If omitted, null, or an empty string, the event data will be transferred unchanged to the provider (full body). |
EventSettingsSerialized | An optional setting where you can set subscription-specific metadata for the provider as details of the event interpretation. This may include some rules, instructions for the provider, etc. For example, if you hypothetically have a workflow provider, you can set what such provider needs to do as a reaction for the event catch: start a new workflow chain or signal an existing workflow instance. The value varies from provider to provider. Please read the provider instruction. |
Events | Array of the event full names the subscription in question should catch. |
Managing Subscriptions through Configuration
Add the subscription array under the key "EventBus:Subscriptions". Subscriptions configured in such a manner cannot be removed or updated through REST API.
If you have subscriptions with the same name in the DB and configuration options, the one specified in the configuration options will be preferred.
Managing Subscriptions through REST API
Get specific subscription by name
Endpoint: /api/eventbus/subscriptions/{name}
Method: GET
Request parameter: Subscription name
Response:
{
"name": "string",
"connectionName": "string",
"jsonPathFilter": "string",
"payloadTransformationTemplate": "string",
"eventSettingsSerialized": "string",
"events": [
{
"eventId": "string"
}
],
"createdDate": "2022-08-29T11:30:09.038Z",
"modifiedDate": "2022-08-29T11:30:09.038Z",
"createdBy": "string", // If null, the subscription specified in the configuration
"modifiedBy": "string", // If null, the subscription specified in the configuration
"id": "string" // If null, the subscription specified in the configuration
}
Registering New Subscription in Database
Endpoint: /api/eventbus/subscriptions
Method: POST
Request body (also check the subscription option description above):
{
"name": "string",
"connectionName": "string",
"jsonPathFilter": "string",
"payloadTransformationTemplate": "string",
"eventSettingsSerialized": "string",
"events": [
{
"eventId": "string"
}
]
}
Updating Existing Subscription (DB Registered Only)
You may want to update the event subscription, so that you could update a set of events.
Endpoint: /api/eventbus/subscriptions
Method: PUT
Request body (also check the subscription option description above):
{
"name": "string",
"connectionName": "string",
"jsonPathFilter": "string",
"payloadTransformationTemplate": "string",
"eventSettingsSerialized": "string",
"events": [
{
"eventId": "string"
}
]
}
Deleting Existing Subscription by Name (DB Registered Only)
You may want to remove the event subscription, so that you could stop receiving event notifications.
Endpoint: /api/eventbus/subscriptions/{name}
Method: DELETE
Viewing List of Subscriptions or Searching for Existing Subscriptions (DB Registered and Configuration Registered)
You may want to see the event subscriptions, so that you know which subscriptions exist.
Endpoint: /api/eventbus/subscriptions/search
Method: POST
Request body:
{
"name": "string", // Subscription name (optional, pass to search by name)
"connectionName": "string", // Provider connection name (optional, pass to search by it)
"eventIds": [ // Optional. Pass to search subscriptions by event ids
"string"
],
"skip": 0, // subscriptions to skip in paged loading
"take": 0 // subscriptions to take in paged loading
}
Response:
{
"totalCount": 0,
"results": [
{
"name": "string",
"connectionName": "string",
"jsonPathFilter": "string",
"payloadTransformationTemplate": "string",
"eventSettingsSerialized": "string",
"events": [
{
"eventId": "string",
}
],
"createdDate": "2022-08-29T11:53:53.653Z",
"modifiedDate": "2022-08-29T11:53:53.653Z",
"createdBy": "string", // If null, the subscription specified in the configuration
"modifiedBy": "string", // If null, the subscription specified in the configuration
"id": "string" // If null, the subscription specified in the configuration
}
]
}
Discovering Current List of Events or Resources
If you want to see the full list of the existing events to properly create a subscription, here is how:
Endpoint: /api/eventbus/events
Method: GET
Request: /api/eventbus/events?skip=0&take=20
Response:
[
{
"id": "VirtoCommerce.CatalogModule.Core.Events.ProductChangedEvent"
},
...
]
Destination Providers
Azure Event Grid Provider
Overview
Azure Event Grid can be used to push messages to Azure Functions, HTTP endpoints (webhooks), and some other Azure tools.
Azure Event Grid supports CloudEvents 1.0, while the Azure Event Grid client library also supports sending and receiving events in the form of CloudEvents.
The Event Bus module contains Azure Event Grid provider, which is ready to use. To connect to it, you need to define the provider connection with the AzureEventGrid
provider name, and fill connection option data structure (the ConnectionOptionsSerialized
field) with the following value:
{
"ConnectionString": "https://*.*.eventgrid.azure.net/api/events",
"AccessKey": "kvpXffggvvMiNjBeBKdroX1r45UvZloXMwlM7i1TyqoiI="
}
connectionString
: String that defines the URI of the topicaccessKey
: String that is partially hidden on retrieval
To set up a subscription with Azure Event Grid, you need first to create a topic in the Azure Portal. When creating your Event Grid topic, you need to set the input schema to CloudEvents v1.0
in the Advanced tab. To allow Virto Commerce Platform to push messages to your topic, you need to provide an access key. These can also be found in the Azure Portal after creating the topic in the Access Keys section.
The EventSettingsSerialized
option of the subscription is not used by this provider and should be skipped.
Error Handling
Event Grid provides durable delivery. It delivers each message at least once for each subscription. Events are sent to the registered endpoint of each subscription immediately. If an endpoint does not acknowledge the receipt of an event, Event Grid retries delivery of the event.
You can find more details about this in Azure Portal.
Default Event Data Model for Azure Event Grid
As mentioned above, you can specify the payload transformation through the Scriban-template with the payloadTransformationTemplate
option.
If you skip this option, the Event Grid provider will apply the following structure as a payload in the CloudEvents
format, as here:
{
"ObjectId": "4038511b-604a-4031-9aba-775bbac43a39",
"ObjectType": "VirtoCommerce.OrdersModule.Core.Model.CustomerOrder",
"EventId": "VirtoCommerce.OrdersModule.Core.Events.OrderChangedEvent"
}
ObjectId
(string type): Object unique identifierObjectType
(string type): Full name of related object typeEventId
(string type): Full name of the Event ID; required
The Event Grid provider discovers all objects related to the event being transferred and generates payloads for each of them.
Sample Event in CloudEvents
1.0 JSON Format
{
"id": "9ec0a767-5789-4149-83ea-bd227570e54a",
"source": "399c9dda-aff9-4bd9-87b4-326dbe2815a9",
"data": {
"ObjectId": "4038511b-604a-4031-9aba-775bbac43a39",
"ObjectType": "VirtoCommerce.OrdersModule.Core.Model.CustomerOrder",
"EventId": "VirtoCommerce.OrdersModule.Core.Events.OrderChangedEvent"
},
"type": "VirtoCommerce.OrdersModule.Core.Events.OrderChangedEvent",
"time": "2021-02-26T08:45:57.3896153Z",
"specversion": "1.0",
"traceparent": "00-22fb7c5208a34c41811cca2715e8d71e-d856ef9e25234f41-00"
}
Health Status and Searching for Fail Log
There is API endpoint that allows you to view the fail log.
Endpoint: /api/eventbus/logs/search
Method: POST
Request:
{
"providerConnectionName": "string", // Optional. Pass to filter the log by provider connection
"startCreatedDate": "2022-08-29T12:42:07.944Z", // Start date of event occurrence
"endCreatedDate": "2022-08-29T12:42:07.944Z", // Start date of event occurrence
"skip": 0, // errors to skip in paged loading
"take": 0 // errors to take in paged loading
}
Response:
{
"totalCount": 0,
"results": [
{
"providerName": "string", // Provider connection name
"status": 0, // Error status
"errorMessage": "string", // Error message
"createdDate": "2022-08-29T12:47:07.793Z", // Date of occurrence
}
]
}
Records in response are always ordered by the date of occurrence, descending.
How to Send Custom Event from Virto Commerce
The module reads the list of the events from installed modules.
If you want to send a new event, you need to create a new module and raise a Virto Commerce event. After this, the event will be accessible via API, and you will be able to create a subscription.
Support for Custom Destination Providers
Feel free to contact us if you need a new destination.
More examples (close to real cases)
Additional event filtering example
Look at the subscription example forwarding order changed event if the state changed to specified only:
"Subscriptions": [
{
"ConnectionName": "AzureEventGrid",
"Name": "Eventgrid forwarder",
"JsonPathFilter": "$.ChangedEntries[?(@.NewEntry.Status == 'Processing' && @.OldEntry.Status != 'Processing')]",
"Events": [
{
"EventId": "VirtoCommerce.OrdersModule.Core.Events.OrderChangedEvent"
}
]
}
]
JsonPathFilter
expression above. The event data will be forwarded to the connection if the selection with specified JsonPathFilter
results any value.
In example: any order comes in status Processing
from any other non-processing state. Another words: we check that new status in the event body is Processing
and old status value is something different.
You can construct more sophisticated expressions for events filtering.
As we use Newtonsoft JsonNet library to select tokens in the json-documents, there are good place to learn JsonPath: Newtonsoft JsonNet documentation: Querying JSON with JSON Path.
Payload transformation template example
The transformation template allows you to transform the event data to a custom payload for your specific case. Also, it is useful to shrink an amount of transferred data. Look at the subscription example:
"Subscriptions": [
{
"ConnectionName": "AzureEventGrid",
"Name": "Eventgrid forwarder",
"JsonPathFilter": "$.ChangedEntries[?(@.NewEntry.Status == 'Processing' && @.OldEntry.Status != 'Processing')]",
"PayloadTransformationTemplate": "{ \"EventId\": \"{{ id }}\", \"OrderInfo\": [ {{for entry in changed_entries}} { \"NewStatus\": \"{{ entry.new_entry.status }}\", \"OldStatus\": \"{{ entry.old_entry.status }}\", \"Items\":[ {{for item in entry.new_entry.items}} { \"Name\": \"{{item.name}}\", \"Sku\": \"{{item.sku}}\" }, {{end}} ] }, {{end}} ] }",
"Events": [
{
"EventId": "VirtoCommerce.OrdersModule.Core.Events.OrderChangedEvent"
}
]
}
]
PayloadTransformationTemplate
value set to some value. It's a one-line, double-comma escaped value of a following Scriban-template:
{
"EventId": "{{ id }}",
"OrderInfo": [
{{for entry in changed_entries}}
{
"NewStatus": "{{ entry.new_entry.status }}",
"OldStatus": "{{ entry.old_entry.status }}",
"Items":[
{{for item in entry.new_entry.items}}
{
"Name": "{{item.name}}",
"Sku": "{{item.sku}}"
},
{{end}}
]
},
{{end}}
]
}
OrderChangedEvent
for some order with 2 items:
{
"EventId": "f90bcd6b-e32b-4d53-9e32-69b9b7fef584",
"OrderInfo": [
{
"NewStatus": "Processing",
"OldStatus": "New",
"Items": [
{
"Name": "Samsung Galaxy S6 SM-G920F 32GB, White Pearl, 1.5 GHz ARM Cortex A53 Quad-Core, 3, 32",
"Sku": "IZZ-25623049"
},
{
"Name": "Microsoft Lumia 640 XL RM-1065 8GB Dual SIM, Black, true, true, 1.2 GHz ARM Cortex A7 Quad-Core, 1, 8",
"Sku": "UWT-27354339"
}
]
}
]
}