Power Automate からプレミアムコネクタなしで Graph API が利用できるので
Graph API を利用して、会議室の予約情報を参照し、iPad などで表示するアプリケーションを Power Apps で作成してみる。
Power Apps の作成
空のキャンバスアプリから作成。
Power Automateの新しいフローを作成する。
Power Automate の作成
全体の流れ
Power Appsから会議室のメールアドレスを引数に会議情報を返却するフローのサンプルを以下に記載する。
詳細については以降で細かく説明。
1. PowerApps (V2)
今回のフローは PowerApps (V2) をトリガーに動作する。
引数として会議室のメールアドレスを受け取るため、その変数名と説明を記載する。
2. 変数を初期化する(返却値)
会議室が使用中の場合も空室の場合も同じ値を返却するためにオブジェクトとして定義する。
正直、文字列変数でもどちらでもいい。
拡張性(例えば、複数の会議室情報を返却する・複数の会議予定を返却する等)を考えるとオブジェクトのほうが良いかと個人的には思う。
3. 現在時刻の書式変更
この後のGraph APIを呼び出す時の時刻書式を変更するためだけに利用。
もう少しいいやり方はあるかもしれない・・・。
4. GraphAPIで会議室の予約状況を取得する
Office 365 Groups を利用して、GraphAPI を呼び出す。ここがこのフローのメイン。
今回は以下のAPIを使用
https://learn.microsoft.com/ja-jp/graph/api/user-list-calendarview?view=graph-rest-1.0
URIの中に設定した値はこんな感じ。
concat('https://graph.microsoft.com/v1.0/users/',triggerBody()['text'],'/calendar/calendarView?startDateTime=',body('現在時刻の書式変更'),'&endDateTime=',body('現在時刻の書式変更'),'&orderby=start/dateTime&top=1')
複数件取れることはないだろうが念のため、開始時間でソート(orderby=start/dateTime)し、最初の1件目(top=1)を取得するようにしている。
各変数にはそれぞれ以下の内容が入る。
変数 | 内容 |
---|---|
triggerBody()['text'] | 1. PowerApps (V2) で取得した引数となるメールアドレス |
body('現在時刻の書式変更') | 3. 現在時刻の書式変更 で変換した現在時刻書式 API仕様上、時刻の書式は ISO 8601 でタイムゾーンは UTC で解釈されるため注意。 |
5. JSON の解析(GraphAPI取得結果)
これが厄介。特にスキーマ情報は公開されていなかったので、Graphエクスプローラー を使いながら微調整。
スキーマ(クリックで開く)
{
"type": "object",
"properties": {
"@@odata.context": {
"type": "string"
},
"value": {
"type": "array",
"items": {
"type": "object",
"properties": {
"@@odata.etag": {
"type": "string"
},
"id": {
"type": "string"
},
"createdDateTime": {
"type": "string"
},
"lastModifiedDateTime": {
"type": "string"
},
"changeKey": {
"type": "string"
},
"categories": {
"type": "array"
},
"transactionId": {},
"originalStartTimeZone": {
"type": "string"
},
"originalEndTimeZone": {
"type": "string"
},
"iCalUId": {
"type": "string"
},
"reminderMinutesBeforeStart": {
"type": "integer"
},
"isReminderOn": {
"type": "boolean"
},
"hasAttachments": {
"type": "boolean"
},
"subject": {
"type": "string"
},
"bodyPreview": {
"type": "string"
},
"importance": {
"type": "string"
},
"sensitivity": {
"type": "string"
},
"isAllDay": {
"type": "boolean"
},
"isCancelled": {
"type": "boolean"
},
"isOrganizer": {
"type": "boolean"
},
"responseRequested": {
"type": "boolean"
},
"seriesMasterId": {},
"showAs": {
"type": "string"
},
"type": {
"type": "string"
},
"webLink": {
"type": "string"
},
"onlineMeetingUrl": {},
"isOnlineMeeting": {
"type": "boolean"
},
"onlineMeetingProvider": {
"type": "string"
},
"allowNewTimeProposals": {
"type": "boolean"
},
"occurrenceId": {},
"isDraft": {
"type": "boolean"
},
"hideAttendees": {
"type": "boolean"
},
"responseStatus": {
"type": "object",
"properties": {
"response": {
"type": "string"
},
"time": {
"type": "string"
}
}
},
"body": {
"type": "object",
"properties": {
"contentType": {
"type": "string"
},
"content": {
"type": "string"
}
}
},
"start": {
"type": "object",
"properties": {
"dateTime": {
"type": "string"
},
"timeZone": {
"type": "string"
}
}
},
"end": {
"type": "object",
"properties": {
"dateTime": {
"type": "string"
},
"timeZone": {
"type": "string"
}
}
},
"location": {
"type": "object",
"properties": {
"displayName": {
"type": "string"
},
"locationType": {
"type": "string"
},
"uniqueIdType": {
"type": "string"
},
"address": {
"type": "object",
"properties": {}
},
"coordinates": {
"type": "object",
"properties": {}
}
}
},
"locations": {
"type": "array"
},
"recurrence": {},
"attendees": {
"type": "array",
"items": {
"type": "object",
"properties": {
"type": {
"type": "string"
},
"status": {
"type": "object",
"properties": {
"response": {
"type": "string"
},
"time": {
"type": "string"
}
}
},
"emailAddress": {
"type": "object",
"properties": {
"name": {
"type": "string"
},
"address": {
"type": "string"
}
}
}
},
"required": [
"type",
"status",
"emailAddress"
]
}
},
"organizer": {
"type": "object",
"properties": {
"emailAddress": {
"type": "object",
"properties": {
"name": {
"type": "string"
},
"address": {
"type": "string"
}
}
}
}
},
"onlineMeeting": {}
},
"required": [
"@@odata.etag",
"id",
"createdDateTime",
"lastModifiedDateTime",
"changeKey",
"categories",
"transactionId",
"originalStartTimeZone",
"originalEndTimeZone",
"iCalUId",
"reminderMinutesBeforeStart",
"isReminderOn",
"hasAttachments",
"subject",
"bodyPreview",
"importance",
"sensitivity",
"isAllDay",
"isCancelled",
"isOrganizer",
"responseRequested",
"seriesMasterId",
"showAs",
"type",
"webLink",
"onlineMeetingUrl",
"isOnlineMeeting",
"onlineMeetingProvider",
"allowNewTimeProposals",
"occurrenceId",
"isDraft",
"hideAttendees",
"responseStatus",
"body",
"start",
"end",
"location",
"locations",
"recurrence",
"attendees",
"organizer",
"onlineMeeting"
]
}
}
}
}
6. 繰り返し(JSON)
JSON の解析(GraphAPI取得結果)で解析した結果から返却値を作成する。
Graph API で取得した結果はUTCなので会議の開催開始時間、終了時間をそれぞれJSTに変換する。
変換後は変数に設定する。
7. JSON の解析(返却値)
スキーマ(クリックで開く)
{
"type": "object",
"properties": {
"MeetingTitle": {
"type": "string"
},
"MeetingTime": {
"type": "string"
},
"MeetingOrganizers": {
"type": "string"
}
}
}
8. PowerApp または Flow に応答する
Power Apps に Graph API で取得した結果を返却する。
Power Apps の変更
OnVisible(画面表示時)に以下を設定。
初期表示時に会議室のメールアドレス設定とメールアドレスを引数に Power Automate を呼び出し、結果を変数に設定。
Set(MeetingRoomAddress, "xxxxx@xxxxx.com");
UpdateContext({Reponse: 'PowerAppV2->[作成した Power Automate のフロー名]'.Run(MeetingRoomAddress)});
Set(MeetingTitle, Reponse.meetingtitle);
Set(MeetingTime, Reponse.meetingtime);
Set(MeetingOrganizers, Reponse.meetingorganizers);
テキストラベルにはレスポンス値が設定される変数をそれぞれ定義してあげればよい。
現時点では画面表示時しか取得できないので例えばタイマーを利用して、数分毎に予約状況を更新すればよい。
他にも複数の会議室を管理している場合は設定用の画面を設けたり、手動の更新ボタンを用意することで実際の使用に繋げられる。