This is the third and last article in which we are going to talk about how to send a JSON with a custom structure through Business Events. In the previous article we saw a possible way to identify those business events that should undergo this new structure through the event catalog. Today, we will finally see how to modify the structure of the message sent for these events.
Modify the structure of the JSON sent through Business Events
The first step is going to be defining the data contract to comply with the JSON format that we need to generate. Let’s remember that the format in question should be something similar to this:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
{ "Id":"06F99161-8C52-4E38-A46F-13918EA48129", "EventTime":"2020-03-24T10:13:36", "EventType":"Project.ERP.JATEventType", "Data": { "BusinessEventId":"JATCustomBusinessEvent", "ControlNumber":5637145326, "EventId":"EA0B3262-EF90-4100-AD0D-3F7E4216A6AB", "EventTime":"/Date(1584617072000)/", "MajorVersion":0, "MinorVersion":0, "JATCustomAttribute":"CustomValue" } } |
Note that the content of the “Data” node is the one that generates a data contract that inherits from the BusinessEventsContract class in a standard way. Therefore, what we will do is generate a data contract with the necessary attributes, with “Data” being an attribute of the JATBusinessEventContract type. (You can get more details about how to generate these objects in this article).
Therefore, our custom data contract should look like this:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 |
[DataContract] class JATCustomDataContract { private str eventType; private str eventTime; private str id; private JATBusinessEventsContract data; [DataMember('Id')] public str parmId(str _id = id) { id = _id; return id; } [DataMember('EventType')] public str parmEventType(str _eventType = eventType) { eventType = _eventType; return eventType; } [DataMember('EventTime')] public str parmEventTime(str _eventTime = eventTime) { eventTime = _eventTime; return eventTime; } [DataMember('Data')] public JATBusinessEventsContract parmData(JATBusinessEventsContract _data = data) { data = _data; return data; } public static JATCustomDataContract construct() { JATCustomDataContract customDataContract = new JATCustomDataContract(); customDataContract.parmEventTime(customDataContract.getDateTimeStr()); return customDataContract; } private str getDateTimeStr() { UtcDateTime dateTimeNow = DateTimeUtil::applyTimeZoneOffset(DateTimeUtil::utcNow(), DateTimeUtil::getCompanyTimeZone()); str dateTimeStr = strReplace(DateTimeUtil::toStr(dateTimeNow), ' ', 'T'); return dateTimeStr; } protected void new() { } } |
Once we have defined our structure, we just need to collect the original message and transform it so that it complies with our format. After spending quite a bit of time debugging and learning how business events work when they are triggered, I reached the conclusion that the easiest and least intrusive way to do this was at the time of insertion into the BusinessEventsCommitLog table.
What the system does is launch the business event, create the message through the data collected in the data contract, serialize the information to generate the JSON, and once everything is ready, insert this information in the table above mentioned. This table is filled with the attributes collected in the BusinessEventsContract class, so we cannot change its structure if we want everything to continue working the same. Once inserted there, a process reads the data, and sends it to the corresponding endpoint.
Therefore, the solution is based on modifying the JSON already generated, so we will not interrupt the standard operation. To achieve that, what we do is extend the insert() method of the BusinessEventsCommitLog table, and just before inserting, we will modify the content of that JSON.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
[ExtensionOf(tableStr(BusinessEventsCommitLog))] final class BusinessEventsCommitLogJATCustomEvent_Extension { public void insert() { BusinessEventsTable businessEventsTable; select firstonly JATCustomEvent, JATEventType, Contract from businessEventsTable where businessEventsTable.BusinessEventId == this.BusinessEventId; if (businessEventsTable.JATCustomEvent) { this.SerializedContract = JATSerializeCustomJsonFormat::getSerializedCustomContract(this.BusinessEventId, businessEventsTable.JATEventType, businessEventsTable.Contract, this.SerializedContract); } next insert(); } } |
With this extension, we will send to generate our personalized format as long as the Event that has been launched is identified within the event catalog as one of our personalized events. If yes, we generate the new JSON as follows:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
class JATSerializeCustomJsonFormat { public static str getSerializedCustomContract(BusinessEventsId _businessEventId, EventTypeDescription _eventType, ClassName _businessEventContractClass, str _serializedContract) { JATCustomDataContract customDataContract; BusinessEventsContract businessEventsContract; str customSerializedContract; // Payload businessEventsContract = FormJsonSerializer::deserializeObject(className2Id(_businessEventContractClass), _serializedContract); // CustomContract customDataContract = JATCustomDataContract::construct(); customDataContract.parmId(businessEventsContract.parmEventId()); customDataContract.parmEventType(_eventType); customDataContract.parmData(businessEventsContract); customSerializedContract = FormJsonSerializer::serializeClass(customDataContract); return customSerializedContract; } } |
Basically, what we do is deserialize the JSON to get an object of the data contract that we are working with (we pass by parameters the name of the data contract to make it reusable), and once we have that object, we encapsulate it within the new data contract, which is re-serialized and is returned by the method to be inserted into the SerializedContract field of the BusinessEventsCommitLog table.
The rest of the process will remain 100% standard, therefore we do not need to extend anything related to the connection or the message handling from Dynamics 365 to the outside, which greatly facilitates the management of events.
With these latest modifications we are already prepared to be able to consume our personalized business events from any of the extremes that we have available.
As a result of these 3 articles, we have managed to generate a non-intrusive and 100% reusable solution. To generate new business events that comply with the same format, we just need to decorate those events with the attribute generated in the previous article.
Here I leave a link to github where you can find the complete solution.
I hope you find it useful, and of course I am open to all kinds of comments, criticisms and improvements. Greetings from home!
1 comment / Add your comment below