One of the biggest limitations of Power Automate Web is that there is no (obvious) functionality for an error handling block like a Try – Catch.
This guide will show you how to implement an error handling block using the Scope step.
Flow Time – Part 1 – Creating the Error Handler
Before we start updating all of our flows with some error handling logic, we will need to create a shared flow that will be called in the event of an error and generate an email notification.
Create a new flow with a manual trigger and the following parameters.
data:image/s3,"s3://crabby-images/451ea/451eaff03bebc3c71e57a8966a070c49bf1f1136" alt=""
- ResultSet will contain the error message, failure step, etc.
- FlowDetails will contain the details (Name, URL) of the flow that failed
- Email is an optional parameter that designates who will receive the email notification. You don’t have to include this if you want to hardcode a distribution.
The following section is entirely optional and depends on your setup. This section declares a variable to hold the final distribution and then sets it to either the optional email parameter or an email defined in an environment variable.
data:image/s3,"s3://crabby-images/7515f/7515f86cca2a2109fd79bdf7cf12da0368080b6d" alt=""
Next lets Select the error message from the ResultSet parameter that was passed in.
data:image/s3,"s3://crabby-images/a70ae/a70aee05ea980765f9a6a7c04792084293f19d6d" alt=""
- From:
@{json(triggerBody()['text'])}
- Step:
@item()?['name']
- Error: This logic will try to the find the value that is most likely to contain the error message.
@if(or(equals(item()?['status'],'Failed'),not(equals(item()?['outputs/body/error/message'],null))),if(not(equals(item()?['error/message'],null)),item()?['error/message'],item()?['outputs/body/error/message']),null)
Next we Filter the output from the select to remove any empty error messages.
data:image/s3,"s3://crabby-images/3d6e7/3d6e7f6a3a5a750179e6898c1f1ec5a847135b34" alt=""
@body('Select')
@item()?['Error']
@null
Now we create an HTML table from our output for email readability.
data:image/s3,"s3://crabby-images/47ef4/47ef47f3b00fd18102b99ef21f59b1848f278800" alt=""
@{body('Filter_array')}
Next we will generate a URL for the flow run so we can link back to it from an email.
data:image/s3,"s3://crabby-images/57b85/57b85e1b869e557fc3d0d41b22d2081eebb05423" alt=""
@{concat('https://us.flow.microsoft.com/manage/environments/', json(triggerBody()['text_1'])['tags']['environmentName'], '/flows/', json(triggerBody()['text_1'])['name'], '/runs/', json(triggerBody()['text_1'])['run']['name'])}
Now we pull out the name of the flow just to make the email nicer.
data:image/s3,"s3://crabby-images/bd2dc/bd2dc3f86acb7cca4103e1607e4bfb41c52461d8" alt=""
@{json(triggerBody()['text_1'])?['tags']?['flowDisplayName']}
We can now send our email as follows. Make sure to switch to the HTML view and paste the entire snippet. I removed the <head> logic so the screenshot wasn’t ridiculous looking.
data:image/s3,"s3://crabby-images/b5960/b5960ee35b7b727a9541b5971bff6e3fa5d5c1eb" alt=""
- To: EmailToUse is the variable that was updated earlier with the email distribution.
- From:
@{outputs('Compose_-_Get_Flow_Name')}
- Body:
<head>
<style type="text/css">
p {
font-family: "Calibri", Arial, Helvetica, sans-serif;
font-size: 14px;
}
table {
border-collapse: collapse;
}
td, th {
border: 1px solid #ddd;
padding: 8px;
text-align: left;
vertical-align:top;
}
tr:nth-child(even) {
background-color: #e3e3e3;
}
tr:hover {
background-color: #ddd;
}
th {
padding-top: 5px;
padding-bottom: 5px;
text-align: center;
background-color: #ededed;
color: black;
}
</style>
</head>
<body>
<p>The <b>@{outputs('Compose_-_Get_Flow_Name')}</b> flow did not complete successfully due to the errors below. <br>
@{body('Create_HTML_table')}
The flow execution can be reviewed <a href='@{outputs('Compose_-_Flow_Run_URL')}'>here</a></p>
</body>
For reference, this HTML will generate an email as follows.
data:image/s3,"s3://crabby-images/140a8/140a8c047d699b48d67fcc51a4ccc7f26f75dca6" alt=""
Finally, we are going to be running this as a child flow that gets called on an error from ALL flows we are creating so we need to add a response step at the end.
IMPORTANT: In ALL response steps, make sure to go into the settings and set to asynchronous. If you do not, flow executions will sometimes duplicate.
data:image/s3,"s3://crabby-images/5f586/5f5867e5c7a11d84efaa1bda5ee3c46e8a5b4421" alt=""
Now you are all done. Make sure to define Run Only Users so that outlook has a defined connection and we are all set to use it from our flows.
Flow Time – Part 2 – How to use the error handler
Add a scope to your flow and label it as you prefer. For standardization purposes, i generally label it as Scope – Try.
data:image/s3,"s3://crabby-images/b4a9e/b4a9e6acee37a46945932afe9e903c60918561ca" alt=""
I’ve added a compose in the middle to generate an error for this post. The formula is here if you are curious: div(10,0)
Inside of the Try block you will be adding all steps that you want to check for a failure on. Unfortunately, due to a system limitation you cannot add variable initialization steps in a Scope so try to use a Set Variable step in the try block if there is some logic that can potentially fail.
Below the Scope – Try add another scope called Scope – Catch which will contain the actions to perform if ANY step in the Scope – Try errors.
data:image/s3,"s3://crabby-images/89d76/89d76ad0e350c20572050dc9238171754ff5fb89" alt=""
Go to the Configure run after setting in the Scope – Catch and check off has failed and has timed out. This will ensure that the actions in the Scope – Catch only execute in the event of a failure.
data:image/s3,"s3://crabby-images/0a0e6/0a0e6f0d597e85042459df49ede5ccc9d8a3578f" alt=""
Now inside of the catch you will 3 steps as follows.
data:image/s3,"s3://crabby-images/c9eb8/c9eb858c9175b4d7d43314a2a0cc6fc4614b263a" alt=""
The Run a Child Flow is configured as per the below to use the flow we created earlier.
data:image/s3,"s3://crabby-images/9fce7/9fce7f417a55270de1a7a9890a92c7fd4daba372" alt=""
- ResultSet:
@{string(result('Scope_-_Try'))}
- FlowDetails:
@{string(workflow())}
- Email: Enter an override if you have this parameter and want to overlay what’s in the flow.
A 500 code response to generate a failure. Dont forget to set to asynch in the settings!
data:image/s3,"s3://crabby-images/c622b/c622bf2fe96daf13a90d5804bc64cdd2532ace7e" alt=""
Finally a terminate step because im paranoid. No idea if this is really needed.
data:image/s3,"s3://crabby-images/ad30b/ad30b33f47c8b4e62976b24be534ab139e1c086d" alt=""
Now that its all together. Congratulations! You now have dynamic error handling. Below is a screenshot of how all your flows should be organized.
data:image/s3,"s3://crabby-images/b5aa0/b5aa01986b8010aaf492d8f2b4c765c1eb15e5a7" alt=""