前回はこちら。
data:image/s3,"s3://crabby-images/66972/66972b43e92522a9047aff0c13358bfd83d33bf8" alt=""
Power Platform ALM パイプラインを使用したソリューションのデプロイを解説|基本的なパイプライン構築
Power Platformの新しい管理ツール『パイプライン』を使う機会があったので、使い方をメモ。今回やること今回は開発環境で構築したソリューションを運用環境(別の環境)へ、『パイプライン』を使用してデプロイしてみる。まずは以下の環境を用...
GitHubへのコミット
前回作成したPower Platformのデプロイ用のパイプラインを拡張し、GitHubにソリューションをコミットしてみる。 基本的には以下の公式の手順と一緒だけど、一部変更が必要。公式
data:image/s3,"s3://crabby-images/0bd4e/0bd4ef1a62d7dadf756a3aa2d439c8db5a3877d9" alt=""
GitHub Actions を使用してパイプラインを拡張する - Power Platform
Power Automate フローから呼び出される GitHub ワークフローを使用して、ソリューションをダウンロード、展開、コミットします。
構築
手順は以下の通り
- EntraIDへのアプリ登録
- Power Platform環境にアプリケーションユーザーの登録
- GitHub:リポジトリ作成→Actionsの作成
- GitHub:アクセストークン作成
- Power Automate構築:GitHubのAPIコール
EntraIDへの登録
まずはEntra IDの登録から。この手順では以下3つの値をメモする。
- クライアントシークレット
- クライアントID(アプリID)
- テナントID
まずはAzure PortalでEntra IDを開き、
data:image/s3,"s3://crabby-images/38012/3801288ea6d3f6eee1e5fe483898936e3f18538d" alt=""
data:image/s3,"s3://crabby-images/38012/3801288ea6d3f6eee1e5fe483898936e3f18538d" alt=""
[アプリの登録]から[新規登録]を選ぶ。
data:image/s3,"s3://crabby-images/50258/502589caa8a68c99ce7116e359972987a564c3e2" alt=""
data:image/s3,"s3://crabby-images/50258/502589caa8a68c99ce7116e359972987a564c3e2" alt=""
適当な名前を付けて、シングルテナントで登録。
data:image/s3,"s3://crabby-images/7122e/7122e4cc2b8738117d3a0a780889f26295b49dd2" alt=""
data:image/s3,"s3://crabby-images/7122e/7122e4cc2b8738117d3a0a780889f26295b49dd2" alt=""
続いて登録したアプリの[APIのアクセス許可]から、[Dynamics CRM]を選んで、
data:image/s3,"s3://crabby-images/32d55/32d5573bebe10c71183f12a8065af43d6e3575bb" alt=""
data:image/s3,"s3://crabby-images/32d55/32d5573bebe10c71183f12a8065af43d6e3575bb" alt=""
「user_impersonation」をチェックし、[アクセス許可の追加]。
data:image/s3,"s3://crabby-images/e4da3/e4da3f6a8c2a8a60a9918995d5bbe1a0dcccf109" alt=""
data:image/s3,"s3://crabby-images/e4da3/e4da3f6a8c2a8a60a9918995d5bbe1a0dcccf109" alt=""
[証明書とシークレット]から[新しいクライアントシークレット]を選び、
data:image/s3,"s3://crabby-images/a6a91/a6a914f0c1b579e6e1a7f33296dab2ad1149260a" alt=""
data:image/s3,"s3://crabby-images/a6a91/a6a914f0c1b579e6e1a7f33296dab2ad1149260a" alt=""
生成された値をメモ。
data:image/s3,"s3://crabby-images/f3ecf/f3ecf1193d9ec71a3484cd0da660dfb6e8926c90" alt=""
data:image/s3,"s3://crabby-images/f3ecf/f3ecf1193d9ec71a3484cd0da660dfb6e8926c90" alt=""
最後に[概要]から[アプリID]と[テナントID]をメモすれば、この手順は終了。
data:image/s3,"s3://crabby-images/13f67/13f67e8f303e8a13969833c303cc0cce73a16644" alt=""
data:image/s3,"s3://crabby-images/13f67/13f67e8f303e8a13969833c303cc0cce73a16644" alt=""
Power Platform環境にアプリケーションユーザーの登録
続いて、Power Platformの環境にアプリケーションユーザーを作成し、GitHubへDataverseへのセキュリティロールを与える。Power Platform管理センターから、パイプラインを構築した環境を選び、[設定]を押す。
data:image/s3,"s3://crabby-images/7e7c9/7e7c90266d25f81fee18ebc0fb1500b736107761" alt=""
data:image/s3,"s3://crabby-images/7e7c9/7e7c90266d25f81fee18ebc0fb1500b736107761" alt=""
[アプリケーションユーザー]を選び、
data:image/s3,"s3://crabby-images/1a9ce/1a9ceadc975cec89672aa92b8fc64b6cd85b2db5" alt=""
data:image/s3,"s3://crabby-images/1a9ce/1a9ceadc975cec89672aa92b8fc64b6cd85b2db5" alt=""
[新規アプリユーザー]から、[アプリの追加]、
data:image/s3,"s3://crabby-images/52f2f/52f2f1b13bc868480d75f8716ab3ec32400dd938" alt=""
data:image/s3,"s3://crabby-images/52f2f/52f2f1b13bc868480d75f8716ab3ec32400dd938" alt=""
前の手順でEntra IDに追加したアプリを選択する。
data:image/s3,"s3://crabby-images/5b365/5b365aafcb0a6bbcfb9d8cb51d51266b17418986" alt=""
data:image/s3,"s3://crabby-images/5b365/5b365aafcb0a6bbcfb9d8cb51d51266b17418986" alt=""
部署(今回はサンプルなのでルート部署)と、セキュリティロールを設定し、[作成]。
※セキュリティロールはSystem Customizerで確実に動作するけど、権限を最低限にしたい場合はカスタムロールの作成を推奨。data:image/s3,"s3://crabby-images/da043/da0433d64ad239dc6a1f902e8d85a6e456eb80e6" alt=""
以上でPower Platformへのアプリケーションユーザーの登録は完了。
※セキュリティロールはSystem Customizerで確実に動作するけど、権限を最低限にしたい場合はカスタムロールの作成を推奨。
data:image/s3,"s3://crabby-images/da043/da0433d64ad239dc6a1f902e8d85a6e456eb80e6" alt=""
GitHub:リポジトリ作成→Actionsの作成
リポジトリ作成
GitHubへ移動し、リポジトリを作成。
data:image/s3,"s3://crabby-images/39f7b/39f7ba624ca8ed07fbd3ddeb27334145a1a82a66" alt=""
data:image/s3,"s3://crabby-images/39f7b/39f7ba624ca8ed07fbd3ddeb27334145a1a82a66" alt=""
Actions設定
リポジトリを作成したら、Actionsから[set up a workflow yourself]を選択。
data:image/s3,"s3://crabby-images/b13ca/b13ca2d5c96864277a6dd96d1ce991e18fbf3508" alt=""
data:image/s3,"s3://crabby-images/b13ca/b13ca2d5c96864277a6dd96d1ce991e18fbf3508" alt=""
そしたら、公式のコードの一部を修正した以下のコードを貼り付け、[Commit changes]を押す。
data:image/s3,"s3://crabby-images/dce6b/dce6b325e7e5053095a944f3445e8a5c877faf6d" alt=""
コードの修正は以下の通り。
data:image/s3,"s3://crabby-images/dce6b/dce6b325e7e5053095a944f3445e8a5c877faf6d" alt=""
# 変更前 # Unpack the solution - name: unpack solution uses: microsoft/powerplatform-actions/unpack-solution@v0 with: solution-file: "${{ github.event.inputs.solution_name }}.zip" solution-folder: "${{ github.event.repository.name }}" solution-type: 'Both' process-canvas-apps: false overwrite-files: true # 変更後(unpack-solutionがv0だと動かなかったので、v1を使用) # Install PAC - name: Install Power Platform Tools uses: microsoft/powerplatform-actions/actions-install@v1 # Unpack the solution - name: unpack solution uses: microsoft/powerplatform-actions/unpack-solution@v1 with: solution-file: "${{ github.event.inputs.solution_name }}.zip" solution-folder: "${{ github.event.repository.name }}" solution-type: 'Both' process-canvas-apps: false overwrite-files: true以下全文。
name: Download, unpack and commit the solution to git run-name: Getting ${{ github.event.inputs.solution_name }} from pipelines host environment and committing on: workflow_dispatch: inputs: artifact_url: description: "The url of the Dataverse record ID for the artifact created by the pipelines (Example: https://[your-env].crm.dynamics.com/api/data/v9.0/deploymentartifacts([your-artifact-id])/artifactfile/$value)." required: true solution_name: description: "Name of the Solution in Dataverse environment" required: true user_name: description: "User name for the commit" required: true source_branch: description: "Branch for the solution commit" required: true target_branch: description: "Branch to create for the solution commit" required: false commit_message: description: "Message to provide for the commit" required: true permissions: contents: write jobs: export-unpack-commit: runs-on: ubuntu-latest steps: - uses: actions/checkout@v3 with: ref: ${{ github.event.inputs.source_branch }} # Commit changes to the existing or new branch - name: create new branch if specified shell: pwsh run: | if('${{ github.event.inputs.target_branch }}' -ne '') { git checkout -b ${{ github.event.inputs.target_branch }} ${{ github.event.inputs.source_branch }} } # Export the solution from the artifact created by pipelines - name: download solution from artifact env: CLIENT_ID: ${{secrets.CLIENT_ID}} TENANT_ID: ${{secrets.TENANT_ID}} CLIENT_SECRET: ${{secrets.CLIENT_SECRET}} shell: pwsh run: | $aadHost = "login.microsoftonline.com" $url = "${{ github.event.inputs.artifact_url }}" $options = [System.StringSplitOptions]::RemoveEmptyEntries $dataverseHost = $url.Split("://", $options)[1].Split("/")[0] $body = @{client_id = $env:CLIENT_ID; client_secret = $env:CLIENT_SECRET; grant_type = "client_credentials"; scope = "https://$dataverseHost/.default"; } $OAuthReq = Invoke-RestMethod -Method Post -Uri "https://$aadHost/$env:TENANT_ID/oauth2/v2.0/token" -Body $body $spnToken = $OAuthReq.access_token $headers = New-Object "System.Collections.Generic.Dictionary[[String],[String]]" $headers.Add("Authorization", "Bearer $spnToken") $headers.Add("Content-Type", "application/json") # Download the managed solution $response = Invoke-RestMethod "${{ github.event.inputs.artifact_url }}" -Method 'GET' -Headers $headers $bytes = [Convert]::FromBase64String($response.value) [IO.File]::WriteAllBytes("${{ github.event.inputs.solution_name }}_managed.zip", $bytes) # Download the unmanaged solution (for now we will need to use string manipulation to get the unmanaged solution URL, until the API provides this value) $unmanaged_artifact_url = "${{ github.event.inputs.artifact_url }}".Replace("artifactfile", "artifactfileunmanaged") $response = Invoke-RestMethod "$unmanaged_artifact_url" -Method 'GET' -Headers $headers $bytes = [Convert]::FromBase64String($response.value) [IO.File]::WriteAllBytes("${{ github.event.inputs.solution_name }}.zip", $bytes) # Install PAC - name: Install Power Platform Tools uses: microsoft/powerplatform-actions/actions-install@v1 # Unpack the solution - name: unpack solution uses: microsoft/powerplatform-actions/unpack-solution@v1 with: solution-file: "${{ github.event.inputs.solution_name }}.zip" solution-folder: "${{ github.event.repository.name }}" solution-type: 'Both' process-canvas-apps: false overwrite-files: true # Commit changes to the existing or new branch - name: commit changes shell: pwsh run: | rm -rf ${{ github.event.inputs.solution_name }}.zip rm -rf ${{ github.event.inputs.solution_name }}_managed.zip git config user.name ${{ github.event.inputs.user_name }} git pull git add --all git commit -am "${{ github.event.inputs.commit_message }}" --allow-empty # Push the committed changes to the source branch - name: push to branch shell: pwsh run: | if('${{ github.event.inputs.target_branch }}' -ne '') { git push origin ${{ github.event.inputs.target_branch }} } else { git push origin ${{ github.event.inputs.source_branch }} }
リポジトリにシークレットを登録
該当リポジトリの[Settings]を開いて、[Secrets and variables]の[Actions]を選択し、[New repository secret]を選択。
data:image/s3,"s3://crabby-images/ce90a/ce90a401c7f6b02b43958e35c21df8a0253b2138" alt=""
data:image/s3,"s3://crabby-images/ce90a/ce90a401c7f6b02b43958e35c21df8a0253b2138" alt=""
ここで、Entra ID設定時にメモした3つの値を設定する
data:image/s3,"s3://crabby-images/cdd45/cdd4513977995ed6466f40fb2ea6727bedc91aef" alt=""
以上で、リポジトリとActionsの設定は完了。
- CLIENT_ID : クライアントID
- CLIENT_SECRET:シークレット
- TENANT_ID:テナントID
data:image/s3,"s3://crabby-images/ac53b/ac53b9eadcb63617e0e9c18f008af22e163b5f35" alt=""
data:image/s3,"s3://crabby-images/bd204/bd2045a3a2a0ae5d8b328549100cea5e157d9b26" alt=""
data:image/s3,"s3://crabby-images/cdd45/cdd4513977995ed6466f40fb2ea6727bedc91aef" alt=""
GitHub:アクセストークンの作成
GitHubのテナント(ユーザー)の設定を開き、[Developer Settings]から[Personal access tokens]の[Fine-grained tokens]を選択。[Generate new token]を押す。
data:image/s3,"s3://crabby-images/f22dc/f22dce7232bbb56196201a2cb922d2e02efb9ac1" alt=""
data:image/s3,"s3://crabby-images/f22dc/f22dce7232bbb56196201a2cb922d2e02efb9ac1" alt=""
適当な名前と、トークンの有効期限を設定、
data:image/s3,"s3://crabby-images/ee488/ee4889747f577bde6e195b980bd6225a510b096f" alt=""
data:image/s3,"s3://crabby-images/ee488/ee4889747f577bde6e195b980bd6225a510b096f" alt=""
Repository accessを先ほど作成したリポジトリに限定し、
data:image/s3,"s3://crabby-images/b2474/b247482bbb134d54e880dfc26f7de96bf36fc12f" alt=""
data:image/s3,"s3://crabby-images/b2474/b247482bbb134d54e880dfc26f7de96bf36fc12f" alt=""
Permissionsで[Actions]と[Contents]を「Read and Write」にする。
data:image/s3,"s3://crabby-images/5c918/5c918d2ef4b3a8b61180a012c8ccf6ccc07be52a" alt=""
data:image/s3,"s3://crabby-images/5c918/5c918d2ef4b3a8b61180a012c8ccf6ccc07be52a" alt=""
[Generate]を押下し、出力されるGitHubのアクセストークンをメモする。
data:image/s3,"s3://crabby-images/32142/3214288836ae4d6f0c3db0cd7ddd165347230146" alt=""
以上でアクセストークンの作成は完了。
data:image/s3,"s3://crabby-images/32142/3214288836ae4d6f0c3db0cd7ddd165347230146" alt=""
Power Automate構築:GitHubのAPIコール
最後にGitHubで作成したActionsを呼び出すPower Automateの構築。全体図はこんな感じ。
data:image/s3,"s3://crabby-images/e63a3/e63a305f0af317ea276373bf4e02f5e3f87977b7" alt=""
data:image/s3,"s3://crabby-images/e63a3/e63a305f0af317ea276373bf4e02f5e3f87977b7" alt=""
まずトリガーは、以下のように設定。
※公式リファレンスに記載のある「OnDeploymentRequested」だと問題が発生する場合があるけど、ひとまずこれで。
data:image/s3,"s3://crabby-images/36152/361527de688c22a55f79f108f758aaa368a043e0" alt=""
※公式リファレンスに記載のある「OnDeploymentRequested」だと問題が発生する場合があるけど、ひとまずこれで。
data:image/s3,"s3://crabby-images/36152/361527de688c22a55f79f108f758aaa368a043e0" alt=""
続いて「IDで行を取得する」アクションで、テーブル「展開ステージの実行」から行を取得。
data:image/s3,"s3://crabby-images/39e31/39e31187c483b1288bbca058766661d874a9bf19" alt=""
data:image/s3,"s3://crabby-images/39e31/39e31187c483b1288bbca058766661d874a9bf19" alt=""
最後にHTTPアクションで、以下のURLにPOSTする。
URL:~~~/repos/【テナント(ユーザー)名】/【リポジトリ名】/actions/workflows/【ワークフローファイル名】/dispatches
ヘッダ:Authorization : Bearer 【Githubアクセストークン】
Bodyは以下の通り。 ※source_branchとtarget_branchは必要に応じて設定を変えてください。
ちょっと長かったけど、以上で設定は完了。
URL:~~~/repos/【テナント(ユーザー)名】/【リポジトリ名】/actions/workflows/【ワークフローファイル名】/dispatches
ヘッダ:Authorization : Bearer 【Githubアクセストークン】
data:image/s3,"s3://crabby-images/7704a/7704aba3242b9b2bfa1f51c6a0ec31dd5163c2e9" alt=""
Bodyは以下の通り。 ※source_branchとtarget_branchは必要に応じて設定を変えてください。
{ "ref": "main", "inputs": { "artifact_url": "@{triggerOutputs()?['body/OutputParameters/ArtifactFileDownloadLink']}", "solution_name": "@{triggerOutputs()?['body/OutputParameters/ArtifactName']}", "user_name": "@{triggerOutputs()?['body/OutputParameters/DeployAsUser']}", "source_branch": "main", "commit_message": "@{outputs('ID_で行を取得する')?['body/deploymentnotes']}" } }
data:image/s3,"s3://crabby-images/35c60/35c60f9b3b2c99d8f51ae49fc041d3a86bc61e5d" alt=""
data:image/s3,"s3://crabby-images/a8684/a8684aa32e6bedb5b268ad96fce9b812310fc3c5" alt=""
data:image/s3,"s3://crabby-images/4bb72/4bb72f92e3b9dfe886c57b340d6f597eb57f8565" alt=""
data:image/s3,"s3://crabby-images/76312/76312b923abfef4c702d30775e95c7816aa96b12" alt=""
動作確認
前回の記事で作成したソリューション内のキャンバスアプリのラベルの内容をv1.0.0.4に変更し、保存→公開する。
data:image/s3,"s3://crabby-images/e44d0/e44d0034f79c1328b9693c0bfead86890e870960" alt=""
data:image/s3,"s3://crabby-images/e44d0/e44d0034f79c1328b9693c0bfead86890e870960" alt=""
公開後、設定したパイプラインを起動すると、
data:image/s3,"s3://crabby-images/a4f8d/a4f8dc5df6f0082666d4e41202076ba479908899" alt=""
data:image/s3,"s3://crabby-images/33227/33227f5e88a2c6ddf122f31d3e1ba5d56fa6a4ef" alt=""
data:image/s3,"s3://crabby-images/a4f8d/a4f8dc5df6f0082666d4e41202076ba479908899" alt=""
Power Automateが動き、GitHubのActionsをコール。
data:image/s3,"s3://crabby-images/cb428/cb428b5050f4641ea52397355d4147308ce7e00e" alt=""
data:image/s3,"s3://crabby-images/cb428/cb428b5050f4641ea52397355d4147308ce7e00e" alt=""
少し待つとGitHubのアクションが成功し、
data:image/s3,"s3://crabby-images/42f9e/42f9ecfec333f842363885ed73d65a3986277e23" alt=""
data:image/s3,"s3://crabby-images/42f9e/42f9ecfec333f842363885ed73d65a3986277e23" alt=""
ソリューションの中身がGitHubに入る。
data:image/s3,"s3://crabby-images/fb744/fb744359bdac3bf2ef679e32693400020156aabf" alt=""
data:image/s3,"s3://crabby-images/fb744/fb744359bdac3bf2ef679e32693400020156aabf" alt=""
今回はキャンバスアプリの中身を変えたので、.msappの内容が無事更新されている。
data:image/s3,"s3://crabby-images/512a8/512a8d09e78f481f58cdb298636a57dccbe1a82c" alt=""
data:image/s3,"s3://crabby-images/512a8/512a8d09e78f481f58cdb298636a57dccbe1a82c" alt=""
Actionsが失敗することもある
全く同じ手順でパイプラインを動かし、
data:image/s3,"s3://crabby-images/2ec09/2ec09e178b682fdb35ef82aad31b0f040fa275dd" alt=""
data:image/s3,"s3://crabby-images/2ec09/2ec09e178b682fdb35ef82aad31b0f040fa275dd" alt=""
Power Automateまで無事動くけど、
data:image/s3,"s3://crabby-images/bbcf9/bbcf96e0c314e0f0452712ce31b2dde7a2b7ac16" alt=""
data:image/s3,"s3://crabby-images/bbcf9/bbcf96e0c314e0f0452712ce31b2dde7a2b7ac16" alt=""
GitHubのActionsが失敗することがある。
data:image/s3,"s3://crabby-images/e4024/e4024e8442fc7913505816a0233a345f830ebbd6" alt=""
エラーメッセージ「No file attachment found for attribute」とある通り、これはGitHubのアクションでマネージドソリューションのzipを持ってこようとしたときに、まだパイプラインのソリューションのzip化が完了していないため、起きていると思われる。data:image/s3,"s3://crabby-images/e4024/e4024e8442fc7913505816a0233a345f830ebbd6" alt=""
ちょっと長くなってしまったので、もしこれが起きた場合の対処方法については次回。
コメント