このブログは、Office 365 Advent Calendar 2021 の 25 日目の記事です。
久々ではありますが Advent Calendar への参加をきっかけに、ゆるりとブログを再開したいと思います。
なおブログ内で掲載する内容は、個人の見解に基づくものです。所属組織を代表するものではありません。予めご了承ください。
はじめに
私は仕事柄ライセンスガイドという公開ドキュメント (PDF) をよく読むのですが、最新版が公開されたか否かは自分で都度ダウンロードして確認するしかありませんでした。これは、私以外のチームメンバも同じでした。
私は「ちょっと面倒だし、効率悪いなー。公開されたら自動的に関係者がアクセス可能な場所にダウンロードされ、Update されたことが Teams に通知されたらいいののにな」という、通称「めんどくさがり」を発動し実装してみることに。
せっかく実装し運用を始めたので、実装について備忘録も兼ねて解説したいと思います。
私と同じように、データの更新を定期確認・ダウンロード・通知といった一連の流れを自動化したいと検討されている方の参考になれば幸いです。
Agenda
このような構成で、解説していきたいと思います。
- 前提条件 / アーキテクチャ
- 必要なツール・ライセンス
- 構成
- 実装詳細
- Teams 編
- PowerShell 編
- Power Automate 編
前提条件 / アーキテクチャ
必要なツール・ライセンス
この記事で紹介する内容を実行するために、必要なツール類は以下の通りです。
- Microsoft 365 (法人向けライセンスを想定しています)
- OneDrive for Business
- Microsoft Teams
- Power Automate (Microsoft 365 付帯のもので結構です)
- Windows 11 (10 でも問題ありませんが、せっかくだから 11 にしよ ?)
- タスクスケジューラ (OS に標準で用意されているものです)
- PowerShell (PowerShell Script を実行できる環境であれば、問題ありません)
- PowerShell の実行ポリシーは、RemoteSigned になっていることを前提としています。
構成
今回は、このようなアーキテクチャで実装しました。
実装詳細
ではさっそく、実装の詳細を見ていきましょう。
Teams 編
これからチームを組織する場合は、Teams でチームとチャネルを作ります。
チームやチャネルを作る権限が無い ? それなら、管理者さんにお願いしましょ !
チームを作る
これから新たにチームを作る場合は、次の手順でチームを作成します。
通知用のチャネルを作る
これから通知先のチャネルを作る場合は、次の手順でチャネルを作成します。
データ格納用フォルダを PC と同期する
この手順は、PC に OneDrife for Business のクライアントを導入しアカウント設定が導入している必要があります。
-
OneDrife for Business クライアントが同期を始めると、同期中のポップアップが表示されるため [閉じる] を選択します。
-
エクスプローラーを開き、[チーム名 - チャネル名] のフォルダが同期状態になっていることを確認します。
後の工程でこのフォルダのパスを利用するため、コピーしておきます。
PowerShell 編
PowerShell Script を作成
詳細は後述しますが、まずは下記 Script をコピーし LicenseGuideDownloader.ps1 というファイル名で保存します。 PowerShell ISE で作成すると編集しやすいのでおススメです。
LicenseGuideDownloader.ps1 サンプル
<#
==============================================================
Script Name : License Guide Download Script
Script File : "C:\_work\LicenseGuideDownloader.ps1"
Script Version : 1.0
ExectionPolicy : RemoteSigned
LastUpdate : 2021/11/24
History :
2021/11/24 Create New
==============================================================
--------------------------------------------------------------
Common Parameter
savePath : Directory path synchronized from SharePoint Documents Library by OneDrive Client
logFilePath : File path for log data
arrayUris : Array of redirection URL ; JP Power Platform, EN Power Platform, JP Dynamics 365, EN Dynamics 365
--------------------------------------------------------------
#>
$savePath = "C:\Users\{Account Name}\{Organization Name}\Biz Apps Team - License Channel\"
$logFilePath = "C:\_work\LicenseGuideDonwloadAutomation.log"
$arrayUris = @("https://go.microsoft.com/fwlink/?LinkId=2085130&clcid=0x411","https://go.microsoft.com/fwlink/?LinkId=2085130&clcid=0x409","https://go.microsoft.com/fwlink/?LinkId=866544&clcid=0x411","https://go.microsoft.com/fwlink/?LinkId=866544&clcid=0x409")
<#
--------------------------------------------------------------
Loop function for file download
respCom : Response of Invoke-WebRequest commandlet
respFileName : File Name of donwload target PDF File
saveFilePath : Full Path of downloaded file
--------------------------------------------------------------
#>
function fncFileDownLoad{
for($i=0; $i -lt $arrayUris.count; $i++){
$respCom = Invoke-WebRequest -Uri $arrayUris[$i]
$respFileName = $respCom.BaseResponse.ResponseUri.LocalPath -replace ".*/",""
$saveFilePath = $savePath + $respFileName
If(-not (Test-Path ($saveFilePath))){
Invoke-WebRequest -Uri $arrayUris[$i] -OutFile $saveFilePath
fncLogging("Download Successed : " + $respFileName)
} else {
fncLogging("No-Update : " + $respFileName)
}
}
}
<#
--------------------------------------------------------------
function for logging
respCom : Response of Invoke-WebRequest commandlet
respFileName : File Name of donwload target PDF File
saveFilePath : Full Path of downloaded file
--------------------------------------------------------------
#>
function fncLogging ($LogMsg) {
If (!(Test-Path -Path $logFilePath)){
New-Item -Path $logFilePath -ItemType File -Force
}
Add-Content $logFilePath ((Get-Date -Format o) + " : " + $LogMsg)
}
<#
--------------------------------------------------------------
Call Function
--------------------------------------------------------------
#>
try{
fncFileDownload
} catch{
fncLogging("Failed to File Download Function")
}
LicenseGuideDownloader.ps1 - First Block 解説
- savePath に、先ほど同期したフォルダのパスを指定します。最後に \ を忘れないようにしましょう。
- logFilePath に、ログファイルのパスを指定します。更新チェック・ダウンロードをログに記録するためのものなので、不要な場合は行頭に # を入れてコメントアウトしてください。
- arrayUris は、ドキュメントが公開されている URL をコンマ区切りで指定します。複数の URL を巡回できるよう、配列として URL を格納しています。
$savePath = "C:\Users\{Account Name}\{Organization Name}\Biz Apps Team - License Channel\"
$logFilePath = "C:\_work\LicenseGuideDonwloadAutomation.log"
$arrayUris = @("https://go.microsoft.com/fwlink/?LinkId=2085130&clcid=0x411","https://go.microsoft.com/fwlink/?LinkId=2085130&clcid=0x409","https://go.microsoft.com/fwlink/?LinkId=866544&clcid=0x411","https://go.microsoft.com/fwlink/?LinkId=866544&clcid=0x409")
LicenseGuideDownloader.ps1
- 1-2 行目 : arrayUris 内の配列数繰り返し処理を function 化しています。
- 3-4 行目 : Invoke-WebRequest で情報を取得し、ファイル名を抽出しています。(ファイル名より前の URL 部分を置換処理で削除しています。)
- 5 行目 : 格納するデータのフルパスを生成しています。
- 6 -10 行目 : 5 行目で生成したフルパス (既存データ) の存在を確認し、存在しなければデータをダウンロードしダウンロードをログに記録しています。存在した場合は、更新が無かった旨をログに記録します。ログに記録する必要がない場合は、行頭に # を入れてコメントアウトしてください。
function fncFileDownLoad{
for($i=0; $i -lt $arrayUris.count; $i++){
$respCom = Invoke-WebRequest -Uri $arrayUris[$i]
$respFileName = $respCom.BaseResponse.ResponseUri.LocalPath -replace ".*/",""
$saveFilePath = $savePath + $respFileName
If(-not (Test-Path ($saveFilePath))){
Invoke-WebRequest -Uri $arrayUris[$i] -OutFile $saveFilePath
fncLogging("Download Successed : " + $respFileName)
} else {
fncLogging("No-Update : " + $respFileName)
}
}
}
LicenseGuideDownloader.ps1 - Third Block 解説
- ログに記録する処理を function 化しています。ログに記録する必要がない場合は、このブロックを削除して構いません。
- 2-3 行目 : ログファイルの有無を確認し、存在していなければファイルを作成しています。
- 5 行目 : 日付とログメッセージ (引数) をログファイルに追記しています。
function fncLogging ($LogMsg) {
If (!(Test-Path -Path $logFilePath)){
New-Item -Path $logFilePath -ItemType File -Force
}
Add-Content $logFilePath ((Get-Date -Format o) + " : " + $LogMsg)
}
LicenseGuideDownloader.ps1 - Fourth Block 解説
- 各機能を呼び出し、実処理を行っています。
- 1-2 行目 : First Block を呼び出し、実処理を行っています。
- 3-4 行目 : First Block の呼び出し時、エラーとなった際にログにエラーを記録しています。
try{
fncFileDownload
} catch{
fncLogging("Failed to File Download Function")
}
タスクスケジューラで定期実行する
PowerShell Script が完成したらタスクスケジューラで定期実行し、更新があれば指定したフォルダにダウンロードします。
-
[Windows] + [R] でファイル名を指定して実行を開き、[taskschd.msc] と入力しタスクスケジューラを起動します。
-
操作タブで、先ほどの PowerShell Script を指定します。各項目は、次のように指定しています。
作成したタスクスケジュールを右クリックし、[実行する] を選択し動作確認をしておくことをお勧めします。
正常動作すると、同期フォルダにデータがダウンロードされていることが確認できます。
Power Automate の動作確認に向け、動作確認が終わったらデータを削除しておきましょう。
Power Automate 編
通知用クラウドフローを作成
Power Automate のクラウドフローを作成し、Teams に通知する設定を行います。これから作成するクラウドフローの全体像は、このようになっています。
-
Power Automate にアクセスし、[作成] - [自動化したクラウドフロー] を選択します。
-
トリガーと最初のアクションを設定します。
-
ケースの設定と、次のアクションを設定します。
-
ファイル名の最初の文字列に応じ、内容の異なるメッセージを Teams に投稿します。
- ケースの次の値と等しいは、"Dynamics" と入力します。(Dynamics 365 ... という文字列の Dynamics の部分で判定しています)
- [アクションの追加] を選択し、[チャットやチャネルにアダプティブカードを投稿する] を選択します。
- 投稿者、投稿先、Team、Channel を画像のように選択します。
- Adaptive Card は、後述の JSON と画像を参考に動的コンテンツを当てはめます。
- ケース 2 も同様に、"Power" と入力します。(Power Apps ... という文字列の Power の部分で判定しています)
- Dynamics の際と同様に、Teams のアクションを追加し設定します。
-
設定が完了したら、クラウドフローをテストします。
Adaptive Card の JSON について
Adaptive Card を作成する際は、Adaptive Card Designer で仕上がりをイメージしながら作成することが可能です。クラウドフローで動的コンテンツを利用する部分は取り除き、設定値を変更してデザインを確認してみてください。
Adaptive Card を簡単に作成いただけるよう、各 Adaptive Card の JSON をサンプルとして掲載しておきます。SharePoint の URL 等、変更が必要な個所にコメントを入れてあります。コピーして該当箇所をお試しいただく環境に合わせて変更しご活用ください。
Adaptive Card - Dynamics JSON
{
"type": "AdaptiveCard",
"body": [
{
"type": "TextBlock",
"size": "Large",
"weight": "Bolder",
"text": "License Guide Updated",
"spacing": "None",
"separator": true,
"isSubtle": false
},
{
"type": "ColumnSet",
"columns": [
{
"type": "Column",
"items": [
{
"type": "TextBlock",
"weight": "Bolder",
"text": "Dynamics 365",
"wrap": true
},
{
"type": "TextBlock",
"spacing": "None",
"weight": "Bolder",
"text": "動的コンテンツから、x-ms-file-name-encoded を選択します。",
"isSubtle": true,
"wrap": true
}
],
],
"width": "auto"
}
],
"style": "default",
"spacing": "None",
"separator": true,
"horizontalAlignment": "Left"
}
],
"actions": [
{
"type": "Action.OpenUrl",
"title": "Open PDF",
"weight": "Bolder",
"url": "トリガーで指定した URL を貼り付け、続けて動的コンテンツから x-ms-file-path-encoded を選択します。"
}
],
"$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
"version": "1.3",
"verticalContentAlignment": "Center"
}
Adaptive Card - Power JSON
{
"type": "AdaptiveCard",
"body": [
{
"type": "TextBlock",
"size": "Large",
"weight": "Bolder",
"text": "License Guide Updated",
"spacing": "None",
"separator": true,
"isSubtle": false
},
{
"type": "ColumnSet",
"columns": [
{
"type": "Column",
"items": [
{
"type": "TextBlock",
"weight": "Bolder",
"text": "Power Apps",
"wrap": true
},
{
"type": "TextBlock",
"spacing": "None",
"weight": "Bolder",
"text": "動的コンテンツから、x-ms-file-name-encoded を選択します。",
"isSubtle": true,
"wrap": true
}
],
"width": "auto"
}
],
"style": "default",
"spacing": "None",
"separator": true,
"horizontalAlignment": "Left"
}
],
"actions": [
{
"type": "Action.OpenUrl",
"title": "Open PDF",
"weight": "Bolder",
"url": "トリガーで指定した URL を貼り付け、続けて動的コンテンツから x-ms-file-path-encoded を選択します。"
}
],
"$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
"version": "1.3",
"verticalContentAlignment": "Center"
}
さいごに
いかがでしたでしょうか。PowerShell や Adaptive Card の部分は、馴染みが無いと難しく感じてしまった方もいらっしゃるでしょうか。今回は誰もがコピペで始められるよう、なるべく詳細に説明を入れました。もし不明点等があれば、Twitter までご連絡ください。可能な限りフォローさせていただきます。