Post

Adaptive Cards to Email through MS Graph (Actionable messages) Part 2

Demonstrating how to send an approval adaptive card (Actionable messages) through MS Graph

Adaptive Cards to Email through MS Graph (Actionable messages) Part 2

Prerequisites

Use Cases / Examples

This blog post will not go to much into the nitty gritty of how it works but will cover some use cases that you might see or have or you can use it to develop your own use cases / adaptive cards your imagination and ideas are the only things setting the limit for you (Aswell as what is possible with adaptive cards of course :) )

1. PTO request

PTO Request

JSON

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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
{
    "type": "AdaptiveCard",
    "version": "1.2",
    "originator": "$($originatorId)",
    "hideOriginalBody": true,
    "`$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "body": [
        {
            "type": "Container",
            "items": [
                {
                    "type": "TextBlock",
                    "text": "$header",
                    "wrap": true
                }
            ]
        },
        {
            "type": "Container",
            "items": [
                {
                    "type": "ColumnSet",
                    "columns": [
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "Image",
                                    "style": "Person",
                                    "url": "$userImage",
                                    "altText": "$userName",
                                    "size": "Small"
                                }
                            ],
                            "width": "auto"
                        },
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "TextBlock",
                                    "weight": "Bolder",
                                    "text": "$userName",
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Requested on the: $date",
                                    "isSubtle": true,
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Reason: $reason",
                                    "isSubtle": true,
                                    "wrap": true
                                }
                            ],
                            "width": "stretch"
                        }
                    ]
                }
            ]
        },
        {
            "type": "FactSet",
            "facts": [
                {
                    "title": "From:",
                    "value": "Untill:"
                },
                {
                    "title": "$fromDate",
                    "value": "$toDate"
                }
            ]
        },
                {
            "type": "ActionSet",
            "actions": [
                {
                    "type": "Action.Http",
                    "title": "Approve",
                    "id": "approve",
                    "iconUrl": "",
                    "method": "POST",
                    "body": "{\"Action\":\"Approve\",\"user\":\"$($userUPN)\"}",
                    "url": "$($httpURL)",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                },
                {
                    "type": "Action.Http",
                    "title": "Reject",
                    "id": "reject",
                    "method": "POST",
                    "body": "{\"Action\":\"Reject\",\"user\":\"$($userUPN)\"}",
                    "url": "$httpURL",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                }
            ]
        }
    ]
}

Powershell script:

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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195

# Originator id
$originatorId = "Your originator id"

# Your url for the logic app saved from earlier the same one used for the `HTTP URL`
$httpURL = "Your logic app url"

# Tenant ID, Client ID, and Client Secret for the MS Graph API
$tenantId = "Your-tenant-id"
$clientId = "Your-client-id"
$clientSecret = "Your-Secret"

# Example values for other variables
$userName = "John Doe"
$userUPN = "John.Doe@contoso.com"
$header = "PTO request from $userName"
$userImage = "https://pbs.twimg.com/profile_images/3647943215/d7f12830b3c17a5a9e4afcc370e3a37e_400x400.jpeg"
$reason = "I need to visit family"
$date = get-date -format "yyyy-MM-dd, HH:mm"
$fromDate = get-date -format "yyyy-MM-dd" -date "2026-01-01"
$toDate = get-date -format "yyyy-MM-dd" -date "2026-01-10"

$params = @{
    message         = @{
        subject      = "$header"
        body         = @{
            contentType = "HTML"
            content     = @"
        <html>
        <head>
            <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
            <script type="application/adaptivecard+json">
{
    "type": "AdaptiveCard",
    "version": "1.2",
    "originator": "$($originatorId)",
    "hideOriginalBody": true,
    "`$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "body": [
        {
            "type": "Container",
            "items": [
                {
                    "type": "TextBlock",
                    "text": "$header",
                    "wrap": true
                }
            ]
        },
        {
            "type": "Container",
            "items": [
                {
                    "type": "ColumnSet",
                    "columns": [
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "Image",
                                    "style": "Person",
                                    "url": "$userImage",
                                    "altText": "$userName",
                                    "size": "Small"
                                }
                            ],
                            "width": "auto"
                        },
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "TextBlock",
                                    "weight": "Bolder",
                                    "text": "$userName",
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Requested on the: $date",
                                    "isSubtle": true,
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Reason: $reason",
                                    "isSubtle": true,
                                    "wrap": true
                                }
                            ],
                            "width": "stretch"
                        }
                    ]
                }
            ]
        },
        {
            "type": "FactSet",
            "facts": [
                {
                    "title": "From:",
                    "value": "Untill:"
                },
                {
                    "title": "$fromDate",
                    "value": "$toDate"
                }
            ]
        },
                {
            "type": "ActionSet",
            "actions": [
                {
                    "type": "Action.Http",
                    "title": "Approve",
                    "id": "approve",
                    "iconUrl": "",
                    "method": "POST",
                    "body": "{\"Action\":\"Approve\",\"user\":\"$($userUPN)\"}",
                    "url": "$($httpURL)",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                },
                {
                    "type": "Action.Http",
                    "title": "Reject",
                    "id": "reject",
                    "method": "POST",
                    "body": "{\"Action\":\"Reject\",\"user\":\"$($userUPN)\"}",
                    "url": "$httpURL",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                }
            ]
        }
    ]
}
        </script>
        </head>
        <p>Please find a new owner for the group and fill it in below and press submit. Thank you ín advance! </p>
        </html>
"@
        }
        toRecipients = @(
            @{
                emailAddress = @{
                    address = "Jane.Doe@contoso.com"
                }
            }
        )
    }
    saveToSentItems = "false"
} | ConvertTo-Json -Depth 10



# Default Token Body
$tokenBody = @{
    Grant_Type    = "client_credentials"
    Scope         = "https://graph.microsoft.com/.default"
    Client_Id     = $clientId
    Client_Secret = $clientSecret
}
# Request a Token
$tokenResponse = Invoke-RestMethod -Uri "https://login.microsoftonline.com/$tenantId/oauth2/v2.0/token" -Method POST -Body $tokenBody

# Setting up the authorization headers
$authHeaders = @{
    "Authorization" = "Bearer $($tokenResponse.access_token)"
    "Content-type" = "application/json"
}

# Graph API BASE URI
$graphApiUri = "https://graph.microsoft.com/v1.0"
$uri = "$graphApiUri/users/{user-id or UPN to send the mail from}/sendMail"
$request = Invoke-RestMethod -Method POST -Uri $uri -Headers $authHeaders -Body $params

2. Access request

Access Request

JSON

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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
{
    "type": "AdaptiveCard",
    "version": "1.2",
    "originator": "$($originatorId)",
    "hideOriginalBody": true,
    "`$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "body": [
        {
            "type": "Container",
            "items": [
                {
                    "type": "TextBlock",
                    "text": "$header",
                    "wrap": true
                }
            ]
        },
        {
            "type": "Container",
            "items": [
                {
                    "type": "ColumnSet",
                    "columns": [
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "Image",
                                    "style": "Person",
                                    "url": "$userImage",
                                    "altText": "$userName",
                                    "size": "Small"
                                }
                            ],
                            "width": "auto"
                        },
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "TextBlock",
                                    "weight": "Bolder",
                                    "text": "$userName",
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Requested on the: $date",
                                    "isSubtle": true,
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Reason: $reason",
                                    "isSubtle": true,
                                    "wrap": true
                                }
                            ],
                            "width": "stretch"
                        }
                    ]
                }
            ]
        },
        {
            "type": "Input.ChoiceSet",
            "choices": $($choiceSet | convertto-json),
            "placeholder": "Placeholder text",
            "style": "expanded",
            "spacing": "Small",
            "isMultiSelect": true,
            "id": "groupSelection"
        },
        {
            "type": "ActionSet",
            "actions": [
                {
                    "type": "Action.Http",
                    "title": "Approve selected",
                    "id": "approve",
                    "iconUrl": "",
                    "method": "POST",
                    "body": "{\"Action\":\"Approve\",\"Approved_groups\":\"\",\"user\":\"$($userUPN)\"}",
                    "url": "$($httpURL)",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                },
                {
                    "type": "Action.Http",
                    "title": "Reject All",
                    "id": "reject",
                    "method": "POST",
                    "body": "{\"Action\":\"Reject\",\"Approved_groups\":\"\",\"user\":\"$($userUPN)\"}",
                    "url": "$httpURL",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                }
            ]
        }
    ]
}

Powershell script:

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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
# Originator id
$originatorId = "Your originator id"

# Your url for the logic app saved from earlier the same one used for the `HTTP URL`
$httpURL = "Your logic app url"

# Tenant ID, Client ID, and Client Secret for the MS Graph API
$tenantId = "Your-tenant-id"
$clientId = "Your-client-id"
$clientSecret = "Your-Secret"

# Example values for other variables
$userName = "John Doe"
$userUPN = "John.Doe@contoso.com"
$header = "Access request from $userName"
$userImage = "https://pbs.twimg.com/profile_images/3647943215/d7f12830b3c17a5a9e4afcc370e3a37e_400x400.jpeg"
$reason = "I am new in the company and need more permissions"
$date = get-date -format "yyyy-MM-dd, HH:mm"


$choiceSet = @(
    @{
        title = "Group Name 1"
        value = "Group1"
    },
    @{
        title = "Group Name 2"
        value = "Group2"
    },
    @{
        title = "Group Name 3"
        value = "Group3"
    },
    @{
        title = "Group Name 4"
        value = "Group4"
    }
)

$params = @{
    message         = @{
        subject      = "$header"
        body         = @{
            contentType = "HTML"
            content     = @"
        <html>
        <head>
            <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
            <script type="application/adaptivecard+json">
{
    "type": "AdaptiveCard",
    "version": "1.2",
    "originator": "$($originatorId)",
    "hideOriginalBody": true,
    "`$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "body": [
        {
            "type": "Container",
            "items": [
                {
                    "type": "TextBlock",
                    "text": "$header",
                    "wrap": true
                }
            ]
        },
        {
            "type": "Container",
            "items": [
                {
                    "type": "ColumnSet",
                    "columns": [
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "Image",
                                    "style": "Person",
                                    "url": "$userImage",
                                    "altText": "$userName",
                                    "size": "Small"
                                }
                            ],
                            "width": "auto"
                        },
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "TextBlock",
                                    "weight": "Bolder",
                                    "text": "$userName",
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Requested on the: $date",
                                    "isSubtle": true,
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Reason: $reason",
                                    "isSubtle": true,
                                    "wrap": true
                                }
                            ],
                            "width": "stretch"
                        }
                    ]
                }
            ]
        },
        {
            "type": "Input.ChoiceSet",
            "choices": $($choiceSet | convertto-json),
            "placeholder": "Placeholder text",
            "style": "expanded",
            "spacing": "Small",
            "isMultiSelect": true,
            "id": "groupSelection"
        },
        {
            "type": "ActionSet",
            "actions": [
                {
                    "type": "Action.Http",
                    "title": "Approve selected",
                    "id": "approve",
                    "iconUrl": "",
                    "method": "POST",
                    "body": "{\"Action\":\"Approve\",\"Approved_groups\":\"\",\"user\":\"$($userUPN)\"}",
                    "url": "$($httpURL)",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                },
                {
                    "type": "Action.Http",
                    "title": "Reject All",
                    "id": "reject",
                    "method": "POST",
                    "body": "{\"Action\":\"Reject\",\"Approved_groups\":\"\",\"user\":\"$($userUPN)\"}",
                    "url": "$httpURL",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                }
            ]
        }
    ]
}
        </script>
        </head>
        <p>Please find a new owner for the group and fill it in below and press submit. Thank you ín advance! </p>
        </html>
"@
        }
        toRecipients = @(
            @{
                emailAddress = @{
                    address = "Jane.Doe@contoso.com"
                }
            }
        )

    }
    saveToSentItems = "false"
} | ConvertTo-Json -Depth 10

# Default Token Body
$tokenBody = @{
    Grant_Type    = "client_credentials"
    Scope         = "https://graph.microsoft.com/.default"
    Client_Id     = $clientId
    Client_Secret = $clientSecret
}
# Request a Token
$tokenResponse = Invoke-RestMethod -Uri "https://login.microsoftonline.com/$tenantId/oauth2/v2.0/token" -Method POST -Body $tokenBody

# Setting up the authorization headers
$authHeaders = @{
    "Authorization" = "Bearer $($tokenResponse.access_token)"
    "Content-type" = "application/json"
}

# Graph API BASE URI
$graphApiUri = "https://graph.microsoft.com/v1.0"
$uri = "$graphApiUri/users/{user-id or UPN to send the mail from}/sendMail"
$request = Invoke-RestMethod -Method POST -Uri $uri -Headers $authHeaders -Body $params

And our body will look like this in our logic app returning only the groups selected which we can continue to work off

1
2
3
4
5
{
  "Action":"Approve",
  "Approved_groups":"Group2,Group4",
  "user":"John.Doe@contoso.com"
}

3. Who should be the new owner/approver?

New Owner

New Owner2

JSON

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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
{
    "type": "AdaptiveCard",
    "version": "1.2",
    "originator": "$($originatorId)",
    "hideOriginalBody": true,
    "`$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "body": [
        {
            "type": "Container",
            "items": [
                {
                    "type": "TextBlock",
                    "text": "$header",
                    "wrap": true
                }
            ]
        },
        {
            "type": "Container",
            "items": [
                {
                    "type": "ColumnSet",
                    "columns": [
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "Image",
                                    "style": "Person",
                                    "url": "$userImage",
                                    "altText": "$userName",
                                    "size": "Small"
                                }
                            ],
                            "width": "auto"
                        },
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "TextBlock",
                                    "weight": "Bolder",
                                    "text": "$userName",
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Requested on the: $date",
                                    "isSubtle": true,
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Reason: $reason",
                                    "isSubtle": true,
                                    "wrap": true
                                }
                            ],
                            "width": "stretch"
                        }
                    ]
                }
            ]
        },
        {
            "type": "FactSet",
            "facts": $($factSet | convertto-json),
            "id": "facts"
        },
        {
            "type": "Input.ChoiceSet",
            "choices": $($choiceSet | convertto-json),
            "placeholder": "Please choose one of your direct reports as the new owner",
            "id": "choice"
        },
        {
            "type": "ActionSet",
            "actions": [
                {
                    "type": "Action.Http",
                    "title": "Sellect new owner",
                    "id": "approve",
                    "iconUrl": "",
                    "method": "POST",
                    "body": "{\"Action\":\"Approve\",\"New_Owner\":\"\",\"Old_Owner\":\"$($userUPN)\",\"Objects\":$($objectsJoinedEscaped)}",
                    "url": "$($httpURL)",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                }
            ]
        }
    ]
}

Powershell script:

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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
# Originator id
$originatorId = "Your originator id"

# Your url for the logic app saved from earlier the same one used for the `HTTP URL`
$httpURL = "Your logic app url"

# Tenant ID, Client ID, and Client Secret for the MS Graph API
$tenantId = "Your-tenant-id"
$clientId = "Your-client-id"
$clientSecret = "Your-Secret"

# Example values for other variables
$userName = "John Doe"
$userUPN = "John.Doe@contoso.com"
$header = "We need a new owner to replace $userName"
$userImage = "https://pbs.twimg.com/profile_images/3647943215/d7f12830b3c17a5a9e4afcc370e3a37e_400x400.jpeg"
$reason = "We can see the employee is set to leave the company on $date and request that you as the manager choose a new person to be responsible for the things the previous employee was set as owner of."
$date = get-date -format "yyyy-MM-dd, HH:mm"


$choiceSet = @(
    @{
        title = "Bob Bob"
        value = "Bob.Bob@contoso.com"
    },
    @{
        title = "Jane Doe"
        value = "Jane.Doe@contoso.com"
    },
    @{
        title = "John Doe"
        value = "John.Doe@contoso.com"
    },
    @{
        title = "Shane Doe"
        value = "Shane.Doe@contoso.com"
    }
)

$factSet = @(
    @{
        title = "Group_Name"
        value = "Group"
    },
    @{
        title = "service_user"
        value = "User"
    },
    @{
        title = "Server_Name"
        value = "Server"
    },
    @{
        title = "Server_Name1"
        value = "Server"
    },
    @{
        title = "Server_Name2"
        value = "Server"
    },
    @{
        title = "Server_Name3"
        value = "Server"
    }
)

# Transform the factSet to create a new array with Type and Name properties
$transformedFactSet = $factSet | ForEach-Object {
    [PSCustomObject]@{
        Type = $_.value
        Name = $_.title
    }
}

# Convert the transformed array to a JSON string needs to be oneline which is why we use -Compress
$objectsJoined = $transformedFactSet | ConvertTo-Json -Compress

# Manually escape the quotes in the JSON string needed for adaptive cards
$objectsJoinedEscaped = $objectsJoined -replace '"', '\"'


$params = @{
    message         = @{
        subject      = "$header"
        body         = @{
            contentType = "HTML"
            content     = @"
        <html>
        <head>
            <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
            <script type="application/adaptivecard+json">
{
    "type": "AdaptiveCard",
    "version": "1.2",
    "originator": "$($originatorId)",
    "hideOriginalBody": true,
    "`$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "body": [
        {
            "type": "Container",
            "items": [
                {
                    "type": "TextBlock",
                    "text": "$header",
                    "wrap": true
                }
            ]
        },
        {
            "type": "Container",
            "items": [
                {
                    "type": "ColumnSet",
                    "columns": [
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "Image",
                                    "style": "Person",
                                    "url": "$userImage",
                                    "altText": "$userName",
                                    "size": "Small"
                                }
                            ],
                            "width": "auto"
                        },
                        {
                            "type": "Column",
                            "items": [
                                {
                                    "type": "TextBlock",
                                    "weight": "Bolder",
                                    "text": "$userName",
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Requested on the: $date",
                                    "isSubtle": true,
                                    "wrap": true
                                },
                                {
                                    "type": "TextBlock",
                                    "spacing": "None",
                                    "text": "Reason: $reason",
                                    "isSubtle": true,
                                    "wrap": true
                                }
                            ],
                            "width": "stretch"
                        }
                    ]
                }
            ]
        },
        {
            "type": "FactSet",
            "facts": $($factSet | convertto-json),
            "id": "facts"
        },
        {
            "type": "Input.ChoiceSet",
            "choices": $($choiceSet | convertto-json),
            "placeholder": "Please choose one of your direct reports as the new owner",
            "id": "choice"
        },
        {
            "type": "ActionSet",
            "actions": [
                {
                    "type": "Action.Http",
                    "title": "Sellect new owner",
                    "id": "approve",
                    "iconUrl": "",
                    "method": "POST",
                    "body": "{\"Action\":\"Approve\",\"New_Owner\":\"\",\"Old_Owner\":\"$($userUPN)\",\"Objects\":$($objectsJoinedEscaped)}",
                    "url": "$($httpURL)",
                    "headers": [
                        {
                            "name": "Authorization",
                            "value": ""
                        },
                        {
                            "name": "Content-type",
                            "value": "text/plain"
                        }
                    ]
                }
            ]
        }
    ]
}
        </script>
        </head>
        <p>Please find a new owner for the group and fill it in below and press submit. Thank you ín advance! </p>
        </html>
"@
        }
        toRecipients = @(
            @{
                emailAddress = @{
                    address = "Jane.Doe@contoso.com"
                }
            }
        )

    }
    saveToSentItems = "false"
} | ConvertTo-Json -Depth 10

# Default Token Body
$tokenBody = @{
    Grant_Type    = "client_credentials"
    Scope         = "https://graph.microsoft.com/.default"
    Client_Id     = $clientId
    Client_Secret = $clientSecret
}
# Request a Token
$tokenResponse = Invoke-RestMethod -Uri "https://login.microsoftonline.com/$tenantId/oauth2/v2.0/token" -Method POST -Body $tokenBody

# Setting up the authorization headers
$authHeaders = @{
    "Authorization" = "Bearer $($tokenResponse.access_token)"
    "Content-type" = "application/json"
}

# Graph API BASE URI
$graphApiUri = "https://graph.microsoft.com/v1.0"
$uri = "$graphApiUri/users/{user-id or UPN to send the mail from}/sendMail"
$request = Invoke-RestMethod -Method POST -Uri $uri -Headers $authHeaders -Body $params

And your body in the logic app will look a bit like this

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
{
  "Action": "Approve",
  "New_Owner": "Jane.Doe@contoso.com",
  "Old_Owner": "John.Doe@contoso.com",
  "Objects":
    [
      { "Type": "Group", "Name": "Group_Name" },
      { "Type": "User", "Name": "service_user" },
      { "Type": "Server", "Name": "Server_Name" },
      { "Type": "Server", "Name": "Server_Name1" },
      { "Type": "Server", "Name": "Server_Name2" },
      { "Type": "Server", "Name": "Server_Name3" },
    ],
}

Conclusion

Combine these use cases together with my last blog post: https://mynster9361.github.io/posts/ActionableMessages/

And you will be well on your way to improve some automation process that you launch to the end users in order to gather infomation and take action on the information gathered straight away instead of going through the collection process your self.

This post is licensed under CC BY 4.0 by the author.